xref: /dpdk/drivers/net/memif/memif_socket.c (revision f62d5262ffa70f1f1f14b42205f94f0d8611a94b)
109c7e63aSJakub Grajciar /* SPDX-License-Identifier: BSD-3-Clause
209c7e63aSJakub Grajciar  * Copyright 2018-2019 Cisco Systems, Inc.  All rights reserved.
309c7e63aSJakub Grajciar  */
409c7e63aSJakub Grajciar 
509c7e63aSJakub Grajciar #include <stdlib.h>
609c7e63aSJakub Grajciar #include <fcntl.h>
709c7e63aSJakub Grajciar #include <unistd.h>
809c7e63aSJakub Grajciar #include <sys/types.h>
909c7e63aSJakub Grajciar #include <sys/socket.h>
1009c7e63aSJakub Grajciar #include <sys/ioctl.h>
1109c7e63aSJakub Grajciar #include <errno.h>
1209c7e63aSJakub Grajciar 
1309c7e63aSJakub Grajciar #include <rte_version.h>
1409c7e63aSJakub Grajciar #include <rte_mbuf.h>
1509c7e63aSJakub Grajciar #include <rte_ether.h>
16df96fd0dSBruce Richardson #include <ethdev_driver.h>
17df96fd0dSBruce Richardson #include <ethdev_vdev.h>
1809c7e63aSJakub Grajciar #include <rte_malloc.h>
1909c7e63aSJakub Grajciar #include <rte_kvargs.h>
204851ef2bSDavid Marchand #include <bus_vdev_driver.h>
2109c7e63aSJakub Grajciar #include <rte_hash.h>
2209c7e63aSJakub Grajciar #include <rte_jhash.h>
2309c7e63aSJakub Grajciar #include <rte_string_fns.h>
2409c7e63aSJakub Grajciar 
2509c7e63aSJakub Grajciar #include "rte_eth_memif.h"
2609c7e63aSJakub Grajciar #include "memif_socket.h"
2709c7e63aSJakub Grajciar 
2809c7e63aSJakub Grajciar static void memif_intr_handler(void *arg);
2909c7e63aSJakub Grajciar 
3009c7e63aSJakub Grajciar static ssize_t
memif_msg_send(int fd,memif_msg_t * msg,int afd)3109c7e63aSJakub Grajciar memif_msg_send(int fd, memif_msg_t *msg, int afd)
3209c7e63aSJakub Grajciar {
3309c7e63aSJakub Grajciar 	struct msghdr mh = { 0 };
3409c7e63aSJakub Grajciar 	struct iovec iov[1];
3509c7e63aSJakub Grajciar 	struct cmsghdr *cmsg;
3609c7e63aSJakub Grajciar 	char ctl[CMSG_SPACE(sizeof(int))];
3709c7e63aSJakub Grajciar 
3809c7e63aSJakub Grajciar 	iov[0].iov_base = msg;
3909c7e63aSJakub Grajciar 	iov[0].iov_len = sizeof(memif_msg_t);
4009c7e63aSJakub Grajciar 	mh.msg_iov = iov;
4109c7e63aSJakub Grajciar 	mh.msg_iovlen = 1;
4209c7e63aSJakub Grajciar 
4309c7e63aSJakub Grajciar 	if (afd > 0) {
4409c7e63aSJakub Grajciar 		memset(&ctl, 0, sizeof(ctl));
4509c7e63aSJakub Grajciar 		mh.msg_control = ctl;
4609c7e63aSJakub Grajciar 		mh.msg_controllen = sizeof(ctl);
4709c7e63aSJakub Grajciar 		cmsg = CMSG_FIRSTHDR(&mh);
4809c7e63aSJakub Grajciar 		cmsg->cmsg_len = CMSG_LEN(sizeof(int));
4909c7e63aSJakub Grajciar 		cmsg->cmsg_level = SOL_SOCKET;
5009c7e63aSJakub Grajciar 		cmsg->cmsg_type = SCM_RIGHTS;
5109c7e63aSJakub Grajciar 		rte_memcpy(CMSG_DATA(cmsg), &afd, sizeof(int));
5209c7e63aSJakub Grajciar 	}
5309c7e63aSJakub Grajciar 
5409c7e63aSJakub Grajciar 	return sendmsg(fd, &mh, 0);
5509c7e63aSJakub Grajciar }
5609c7e63aSJakub Grajciar 
5709c7e63aSJakub Grajciar static int
memif_msg_send_from_queue(struct memif_control_channel * cc)5809c7e63aSJakub Grajciar memif_msg_send_from_queue(struct memif_control_channel *cc)
5909c7e63aSJakub Grajciar {
6009c7e63aSJakub Grajciar 	ssize_t size;
6109c7e63aSJakub Grajciar 	int ret = 0;
6209c7e63aSJakub Grajciar 	struct memif_msg_queue_elt *e;
6309c7e63aSJakub Grajciar 
6409c7e63aSJakub Grajciar 	e = TAILQ_FIRST(&cc->msg_queue);
6509c7e63aSJakub Grajciar 	if (e == NULL)
6609c7e63aSJakub Grajciar 		return 0;
6709c7e63aSJakub Grajciar 
68aedd054cSHarman Kalra 	if (rte_intr_fd_get(cc->intr_handle) < 0)
69aedd054cSHarman Kalra 		return -1;
70aedd054cSHarman Kalra 
71d61138d4SHarman Kalra 	size = memif_msg_send(rte_intr_fd_get(cc->intr_handle), &e->msg,
72d61138d4SHarman Kalra 			      e->fd);
7309c7e63aSJakub Grajciar 	if (size != sizeof(memif_msg_t)) {
7409c7e63aSJakub Grajciar 		MIF_LOG(ERR, "sendmsg fail: %s.", strerror(errno));
7509c7e63aSJakub Grajciar 		ret = -1;
7609c7e63aSJakub Grajciar 	} else {
7709c7e63aSJakub Grajciar 		MIF_LOG(DEBUG, "Sent msg type %u.", e->msg.type);
7809c7e63aSJakub Grajciar 	}
7909c7e63aSJakub Grajciar 	TAILQ_REMOVE(&cc->msg_queue, e, next);
8009c7e63aSJakub Grajciar 	rte_free(e);
8109c7e63aSJakub Grajciar 
8209c7e63aSJakub Grajciar 	return ret;
8309c7e63aSJakub Grajciar }
8409c7e63aSJakub Grajciar 
8509c7e63aSJakub Grajciar static struct memif_msg_queue_elt *
memif_msg_enq(struct memif_control_channel * cc)8609c7e63aSJakub Grajciar memif_msg_enq(struct memif_control_channel *cc)
8709c7e63aSJakub Grajciar {
8809c7e63aSJakub Grajciar 	struct memif_msg_queue_elt *e;
8909c7e63aSJakub Grajciar 
9009c7e63aSJakub Grajciar 	e = rte_zmalloc("memif_msg", sizeof(struct memif_msg_queue_elt), 0);
9109c7e63aSJakub Grajciar 	if (e == NULL) {
9209c7e63aSJakub Grajciar 		MIF_LOG(ERR, "Failed to allocate control message.");
9309c7e63aSJakub Grajciar 		return NULL;
9409c7e63aSJakub Grajciar 	}
9509c7e63aSJakub Grajciar 
9609c7e63aSJakub Grajciar 	e->fd = -1;
9709c7e63aSJakub Grajciar 	TAILQ_INSERT_TAIL(&cc->msg_queue, e, next);
9809c7e63aSJakub Grajciar 
9909c7e63aSJakub Grajciar 	return e;
10009c7e63aSJakub Grajciar }
10109c7e63aSJakub Grajciar 
10209c7e63aSJakub Grajciar void
memif_msg_enq_disconnect(struct memif_control_channel * cc,const char * reason,int err_code)10309c7e63aSJakub Grajciar memif_msg_enq_disconnect(struct memif_control_channel *cc, const char *reason,
10409c7e63aSJakub Grajciar 			 int err_code)
10509c7e63aSJakub Grajciar {
10609c7e63aSJakub Grajciar 	struct memif_msg_queue_elt *e;
10709c7e63aSJakub Grajciar 	struct pmd_internals *pmd;
10809c7e63aSJakub Grajciar 	memif_msg_disconnect_t *d;
10909c7e63aSJakub Grajciar 
11009c7e63aSJakub Grajciar 	if (cc == NULL) {
11109c7e63aSJakub Grajciar 		MIF_LOG(DEBUG, "Missing control channel.");
11209c7e63aSJakub Grajciar 		return;
11309c7e63aSJakub Grajciar 	}
11409c7e63aSJakub Grajciar 
11509c7e63aSJakub Grajciar 	e = memif_msg_enq(cc);
11609c7e63aSJakub Grajciar 	if (e == NULL) {
11709c7e63aSJakub Grajciar 		MIF_LOG(WARNING, "Failed to enqueue disconnect message.");
11809c7e63aSJakub Grajciar 		return;
11909c7e63aSJakub Grajciar 	}
12009c7e63aSJakub Grajciar 
12109c7e63aSJakub Grajciar 	d = &e->msg.disconnect;
12209c7e63aSJakub Grajciar 
12309c7e63aSJakub Grajciar 	e->msg.type = MEMIF_MSG_TYPE_DISCONNECT;
12409c7e63aSJakub Grajciar 	d->code = err_code;
12509c7e63aSJakub Grajciar 
12609c7e63aSJakub Grajciar 	if (reason != NULL) {
12709c7e63aSJakub Grajciar 		strlcpy((char *)d->string, reason, sizeof(d->string));
12809c7e63aSJakub Grajciar 		if (cc->dev != NULL) {
12909c7e63aSJakub Grajciar 			pmd = cc->dev->data->dev_private;
13009c7e63aSJakub Grajciar 			strlcpy(pmd->local_disc_string, reason,
13109c7e63aSJakub Grajciar 				sizeof(pmd->local_disc_string));
13209c7e63aSJakub Grajciar 		}
13309c7e63aSJakub Grajciar 	}
13409c7e63aSJakub Grajciar }
13509c7e63aSJakub Grajciar 
13609c7e63aSJakub Grajciar static int
memif_msg_enq_hello(struct memif_control_channel * cc)13709c7e63aSJakub Grajciar memif_msg_enq_hello(struct memif_control_channel *cc)
13809c7e63aSJakub Grajciar {
13909c7e63aSJakub Grajciar 	struct memif_msg_queue_elt *e = memif_msg_enq(cc);
14009c7e63aSJakub Grajciar 	memif_msg_hello_t *h;
14109c7e63aSJakub Grajciar 
14209c7e63aSJakub Grajciar 	if (e == NULL)
14309c7e63aSJakub Grajciar 		return -1;
14409c7e63aSJakub Grajciar 
14509c7e63aSJakub Grajciar 	h = &e->msg.hello;
14609c7e63aSJakub Grajciar 
14709c7e63aSJakub Grajciar 	e->msg.type = MEMIF_MSG_TYPE_HELLO;
14809c7e63aSJakub Grajciar 	h->min_version = MEMIF_VERSION;
14909c7e63aSJakub Grajciar 	h->max_version = MEMIF_VERSION;
150d250589dSStephen Hemminger 	h->max_c2s_ring = ETH_MEMIF_MAX_NUM_Q_PAIRS;
151d250589dSStephen Hemminger 	h->max_s2c_ring = ETH_MEMIF_MAX_NUM_Q_PAIRS;
15209c7e63aSJakub Grajciar 	h->max_region = ETH_MEMIF_MAX_REGION_NUM - 1;
15309c7e63aSJakub Grajciar 	h->max_log2_ring_size = ETH_MEMIF_MAX_LOG2_RING_SIZE;
15409c7e63aSJakub Grajciar 
15509c7e63aSJakub Grajciar 	strlcpy((char *)h->name, rte_version(), sizeof(h->name));
15609c7e63aSJakub Grajciar 
15709c7e63aSJakub Grajciar 	return 0;
15809c7e63aSJakub Grajciar }
15909c7e63aSJakub Grajciar 
16009c7e63aSJakub Grajciar static int
memif_msg_receive_hello(struct rte_eth_dev * dev,memif_msg_t * msg)16109c7e63aSJakub Grajciar memif_msg_receive_hello(struct rte_eth_dev *dev, memif_msg_t *msg)
16209c7e63aSJakub Grajciar {
16309c7e63aSJakub Grajciar 	struct pmd_internals *pmd = dev->data->dev_private;
16409c7e63aSJakub Grajciar 	memif_msg_hello_t *h = &msg->hello;
16509c7e63aSJakub Grajciar 
16609c7e63aSJakub Grajciar 	if (h->min_version > MEMIF_VERSION || h->max_version < MEMIF_VERSION) {
16709c7e63aSJakub Grajciar 		memif_msg_enq_disconnect(pmd->cc, "Incompatible memif version", 0);
16809c7e63aSJakub Grajciar 		return -1;
16909c7e63aSJakub Grajciar 	}
17009c7e63aSJakub Grajciar 
17109c7e63aSJakub Grajciar 	/* Set parameters for active connection */
172d250589dSStephen Hemminger 	pmd->run.num_c2s_rings = RTE_MIN(h->max_c2s_ring + 1,
173d250589dSStephen Hemminger 					   pmd->cfg.num_c2s_rings);
174d250589dSStephen Hemminger 	pmd->run.num_s2c_rings = RTE_MIN(h->max_s2c_ring + 1,
175d250589dSStephen Hemminger 					   pmd->cfg.num_s2c_rings);
17609c7e63aSJakub Grajciar 	pmd->run.log2_ring_size = RTE_MIN(h->max_log2_ring_size,
17709c7e63aSJakub Grajciar 					    pmd->cfg.log2_ring_size);
17809c7e63aSJakub Grajciar 	pmd->run.pkt_buffer_size = pmd->cfg.pkt_buffer_size;
17909c7e63aSJakub Grajciar 
18009c7e63aSJakub Grajciar 	strlcpy(pmd->remote_name, (char *)h->name, sizeof(pmd->remote_name));
18109c7e63aSJakub Grajciar 
18243b815d8SJakub Grajciar 	MIF_LOG(DEBUG, "Connecting to %s.", pmd->remote_name);
18309c7e63aSJakub Grajciar 
18409c7e63aSJakub Grajciar 	return 0;
18509c7e63aSJakub Grajciar }
18609c7e63aSJakub Grajciar 
18709c7e63aSJakub Grajciar static int
memif_msg_receive_init(struct memif_control_channel * cc,memif_msg_t * msg)18809c7e63aSJakub Grajciar memif_msg_receive_init(struct memif_control_channel *cc, memif_msg_t *msg)
18909c7e63aSJakub Grajciar {
19009c7e63aSJakub Grajciar 	memif_msg_init_t *i = &msg->init;
19109c7e63aSJakub Grajciar 	struct memif_socket_dev_list_elt *elt;
19209c7e63aSJakub Grajciar 	struct pmd_internals *pmd;
19309c7e63aSJakub Grajciar 	struct rte_eth_dev *dev;
19409c7e63aSJakub Grajciar 
19509c7e63aSJakub Grajciar 	if (i->version != MEMIF_VERSION) {
19609c7e63aSJakub Grajciar 		memif_msg_enq_disconnect(cc, "Incompatible memif version", 0);
19709c7e63aSJakub Grajciar 		return -1;
19809c7e63aSJakub Grajciar 	}
19909c7e63aSJakub Grajciar 
20009c7e63aSJakub Grajciar 	if (cc->socket == NULL) {
20109c7e63aSJakub Grajciar 		memif_msg_enq_disconnect(cc, "Device error", 0);
20209c7e63aSJakub Grajciar 		return -1;
20309c7e63aSJakub Grajciar 	}
20409c7e63aSJakub Grajciar 
20509c7e63aSJakub Grajciar 	/* Find device with requested ID */
20609c7e63aSJakub Grajciar 	TAILQ_FOREACH(elt, &cc->socket->dev_queue, next) {
20709c7e63aSJakub Grajciar 		dev = elt->dev;
20809c7e63aSJakub Grajciar 		pmd = dev->data->dev_private;
20909c7e63aSJakub Grajciar 		if (((pmd->flags & ETH_MEMIF_FLAG_DISABLED) == 0) &&
210d250589dSStephen Hemminger 		    (pmd->id == i->id) && (pmd->role == MEMIF_ROLE_SERVER)) {
21106123fddSJúlius Milan 			if (pmd->flags & (ETH_MEMIF_FLAG_CONNECTING |
21206123fddSJúlius Milan 					   ETH_MEMIF_FLAG_CONNECTED)) {
21306123fddSJúlius Milan 				memif_msg_enq_disconnect(cc,
21406123fddSJúlius Milan 							 "Already connected", 0);
21506123fddSJúlius Milan 				return -1;
21606123fddSJúlius Milan 			}
21706123fddSJúlius Milan 
21809c7e63aSJakub Grajciar 			/* assign control channel to device */
21909c7e63aSJakub Grajciar 			cc->dev = dev;
22009c7e63aSJakub Grajciar 			pmd->cc = cc;
22109c7e63aSJakub Grajciar 
22209c7e63aSJakub Grajciar 			if (i->mode != MEMIF_INTERFACE_MODE_ETHERNET) {
22309c7e63aSJakub Grajciar 				memif_msg_enq_disconnect(pmd->cc,
22409c7e63aSJakub Grajciar 							 "Only ethernet mode supported",
22509c7e63aSJakub Grajciar 							 0);
22609c7e63aSJakub Grajciar 				return -1;
22709c7e63aSJakub Grajciar 			}
22809c7e63aSJakub Grajciar 
22909c7e63aSJakub Grajciar 			strlcpy(pmd->remote_name, (char *)i->name,
23009c7e63aSJakub Grajciar 				sizeof(pmd->remote_name));
23109c7e63aSJakub Grajciar 
23209c7e63aSJakub Grajciar 			if (*pmd->secret != '\0') {
23309c7e63aSJakub Grajciar 				if (*i->secret == '\0') {
23409c7e63aSJakub Grajciar 					memif_msg_enq_disconnect(pmd->cc,
23509c7e63aSJakub Grajciar 								 "Secret required", 0);
23609c7e63aSJakub Grajciar 					return -1;
23709c7e63aSJakub Grajciar 				}
23809c7e63aSJakub Grajciar 				if (strncmp(pmd->secret, (char *)i->secret,
23909c7e63aSJakub Grajciar 						ETH_MEMIF_SECRET_SIZE) != 0) {
24009c7e63aSJakub Grajciar 					memif_msg_enq_disconnect(pmd->cc,
24109c7e63aSJakub Grajciar 								 "Incorrect secret", 0);
24209c7e63aSJakub Grajciar 					return -1;
24309c7e63aSJakub Grajciar 				}
24409c7e63aSJakub Grajciar 			}
24509c7e63aSJakub Grajciar 
24609c7e63aSJakub Grajciar 			pmd->flags |= ETH_MEMIF_FLAG_CONNECTING;
24709c7e63aSJakub Grajciar 			return 0;
24809c7e63aSJakub Grajciar 		}
24909c7e63aSJakub Grajciar 	}
25009c7e63aSJakub Grajciar 
25109c7e63aSJakub Grajciar 	/* ID not found on this socket */
25209c7e63aSJakub Grajciar 	MIF_LOG(DEBUG, "ID %u not found.", i->id);
25309c7e63aSJakub Grajciar 	memif_msg_enq_disconnect(cc, "ID not found", 0);
25409c7e63aSJakub Grajciar 	return -1;
25509c7e63aSJakub Grajciar }
25609c7e63aSJakub Grajciar 
25709c7e63aSJakub Grajciar static int
memif_msg_receive_add_region(struct rte_eth_dev * dev,memif_msg_t * msg,int fd)25809c7e63aSJakub Grajciar memif_msg_receive_add_region(struct rte_eth_dev *dev, memif_msg_t *msg,
25909c7e63aSJakub Grajciar 			     int fd)
26009c7e63aSJakub Grajciar {
26109c7e63aSJakub Grajciar 	struct pmd_internals *pmd = dev->data->dev_private;
262c41a0495SJakub Grajciar 	struct pmd_process_private *proc_private = dev->process_private;
26309c7e63aSJakub Grajciar 	memif_msg_add_region_t *ar = &msg->add_region;
26409c7e63aSJakub Grajciar 	struct memif_region *r;
26509c7e63aSJakub Grajciar 
26609c7e63aSJakub Grajciar 	if (fd < 0) {
26709c7e63aSJakub Grajciar 		memif_msg_enq_disconnect(pmd->cc, "Missing region fd", 0);
26809c7e63aSJakub Grajciar 		return -1;
26909c7e63aSJakub Grajciar 	}
27009c7e63aSJakub Grajciar 
271c41a0495SJakub Grajciar 	if (ar->index >= ETH_MEMIF_MAX_REGION_NUM ||
272c41a0495SJakub Grajciar 			ar->index != proc_private->regions_num ||
273c41a0495SJakub Grajciar 			proc_private->regions[ar->index] != NULL) {
27409c7e63aSJakub Grajciar 		memif_msg_enq_disconnect(pmd->cc, "Invalid region index", 0);
27509c7e63aSJakub Grajciar 		return -1;
27609c7e63aSJakub Grajciar 	}
27709c7e63aSJakub Grajciar 
27809c7e63aSJakub Grajciar 	r = rte_zmalloc("region", sizeof(struct memif_region), 0);
27909c7e63aSJakub Grajciar 	if (r == NULL) {
280c41a0495SJakub Grajciar 		memif_msg_enq_disconnect(pmd->cc, "Failed to alloc memif region.", 0);
28109c7e63aSJakub Grajciar 		return -ENOMEM;
28209c7e63aSJakub Grajciar 	}
28309c7e63aSJakub Grajciar 
28409c7e63aSJakub Grajciar 	r->fd = fd;
28509c7e63aSJakub Grajciar 	r->region_size = ar->size;
28609c7e63aSJakub Grajciar 	r->addr = NULL;
28709c7e63aSJakub Grajciar 
288c41a0495SJakub Grajciar 	proc_private->regions[ar->index] = r;
289c41a0495SJakub Grajciar 	proc_private->regions_num++;
29009c7e63aSJakub Grajciar 
29109c7e63aSJakub Grajciar 	return 0;
29209c7e63aSJakub Grajciar }
29309c7e63aSJakub Grajciar 
29409c7e63aSJakub Grajciar static int
memif_msg_receive_add_ring(struct rte_eth_dev * dev,memif_msg_t * msg,int fd)29509c7e63aSJakub Grajciar memif_msg_receive_add_ring(struct rte_eth_dev *dev, memif_msg_t *msg, int fd)
29609c7e63aSJakub Grajciar {
29709c7e63aSJakub Grajciar 	struct pmd_internals *pmd = dev->data->dev_private;
29809c7e63aSJakub Grajciar 	memif_msg_add_ring_t *ar = &msg->add_ring;
29909c7e63aSJakub Grajciar 	struct memif_queue *mq;
30009c7e63aSJakub Grajciar 
30109c7e63aSJakub Grajciar 	if (fd < 0) {
30209c7e63aSJakub Grajciar 		memif_msg_enq_disconnect(pmd->cc, "Missing interrupt fd", 0);
30309c7e63aSJakub Grajciar 		return -1;
30409c7e63aSJakub Grajciar 	}
30509c7e63aSJakub Grajciar 
30609c7e63aSJakub Grajciar 	/* check if we have enough queues */
307d250589dSStephen Hemminger 	if (ar->flags & MEMIF_MSG_ADD_RING_FLAG_C2S) {
308d250589dSStephen Hemminger 		if (ar->index >= pmd->cfg.num_c2s_rings) {
30909c7e63aSJakub Grajciar 			memif_msg_enq_disconnect(pmd->cc, "Invalid ring index", 0);
31009c7e63aSJakub Grajciar 			return -1;
31109c7e63aSJakub Grajciar 		}
312d250589dSStephen Hemminger 		pmd->run.num_c2s_rings++;
31309c7e63aSJakub Grajciar 	} else {
314d250589dSStephen Hemminger 		if (ar->index >= pmd->cfg.num_s2c_rings) {
31509c7e63aSJakub Grajciar 			memif_msg_enq_disconnect(pmd->cc, "Invalid ring index", 0);
31609c7e63aSJakub Grajciar 			return -1;
31709c7e63aSJakub Grajciar 		}
318d250589dSStephen Hemminger 		pmd->run.num_s2c_rings++;
31909c7e63aSJakub Grajciar 	}
32009c7e63aSJakub Grajciar 
321d250589dSStephen Hemminger 	mq = (ar->flags & MEMIF_MSG_ADD_RING_FLAG_C2S) ?
32209c7e63aSJakub Grajciar 	    dev->data->rx_queues[ar->index] : dev->data->tx_queues[ar->index];
32309c7e63aSJakub Grajciar 
324d61138d4SHarman Kalra 	if (rte_intr_fd_set(mq->intr_handle, fd))
325d61138d4SHarman Kalra 		return -1;
326d61138d4SHarman Kalra 
32709c7e63aSJakub Grajciar 	mq->log2_ring_size = ar->log2_ring_size;
32809c7e63aSJakub Grajciar 	mq->region = ar->region;
32909c7e63aSJakub Grajciar 	mq->ring_offset = ar->offset;
33009c7e63aSJakub Grajciar 
33109c7e63aSJakub Grajciar 	return 0;
33209c7e63aSJakub Grajciar }
33309c7e63aSJakub Grajciar 
33409c7e63aSJakub Grajciar static int
memif_msg_receive_connect(struct rte_eth_dev * dev,memif_msg_t * msg)33509c7e63aSJakub Grajciar memif_msg_receive_connect(struct rte_eth_dev *dev, memif_msg_t *msg)
33609c7e63aSJakub Grajciar {
33709c7e63aSJakub Grajciar 	struct pmd_internals *pmd = dev->data->dev_private;
33809c7e63aSJakub Grajciar 	memif_msg_connect_t *c = &msg->connect;
33909c7e63aSJakub Grajciar 	int ret;
34009c7e63aSJakub Grajciar 
34109c7e63aSJakub Grajciar 	ret = memif_connect(dev);
34209c7e63aSJakub Grajciar 	if (ret < 0)
34309c7e63aSJakub Grajciar 		return ret;
34409c7e63aSJakub Grajciar 
34509c7e63aSJakub Grajciar 	strlcpy(pmd->remote_if_name, (char *)c->if_name,
34609c7e63aSJakub Grajciar 		sizeof(pmd->remote_if_name));
34743b815d8SJakub Grajciar 	MIF_LOG(INFO, "Remote interface %s connected.", pmd->remote_if_name);
34809c7e63aSJakub Grajciar 
34909c7e63aSJakub Grajciar 	return 0;
35009c7e63aSJakub Grajciar }
35109c7e63aSJakub Grajciar 
35209c7e63aSJakub Grajciar static int
memif_msg_receive_connected(struct rte_eth_dev * dev,memif_msg_t * msg)35309c7e63aSJakub Grajciar memif_msg_receive_connected(struct rte_eth_dev *dev, memif_msg_t *msg)
35409c7e63aSJakub Grajciar {
35509c7e63aSJakub Grajciar 	struct pmd_internals *pmd = dev->data->dev_private;
35609c7e63aSJakub Grajciar 	memif_msg_connected_t *c = &msg->connected;
35709c7e63aSJakub Grajciar 	int ret;
35809c7e63aSJakub Grajciar 
35909c7e63aSJakub Grajciar 	ret = memif_connect(dev);
36009c7e63aSJakub Grajciar 	if (ret < 0)
36109c7e63aSJakub Grajciar 		return ret;
36209c7e63aSJakub Grajciar 
36309c7e63aSJakub Grajciar 	strlcpy(pmd->remote_if_name, (char *)c->if_name,
36409c7e63aSJakub Grajciar 		sizeof(pmd->remote_if_name));
36543b815d8SJakub Grajciar 	MIF_LOG(INFO, "Remote interface %s connected.", pmd->remote_if_name);
36609c7e63aSJakub Grajciar 
36709c7e63aSJakub Grajciar 	return 0;
36809c7e63aSJakub Grajciar }
36909c7e63aSJakub Grajciar 
37009c7e63aSJakub Grajciar static int
memif_msg_receive_disconnect(struct rte_eth_dev * dev,memif_msg_t * msg)37109c7e63aSJakub Grajciar memif_msg_receive_disconnect(struct rte_eth_dev *dev, memif_msg_t *msg)
37209c7e63aSJakub Grajciar {
37309c7e63aSJakub Grajciar 	struct pmd_internals *pmd = dev->data->dev_private;
37409c7e63aSJakub Grajciar 	memif_msg_disconnect_t *d = &msg->disconnect;
37509c7e63aSJakub Grajciar 
37643b815d8SJakub Grajciar 	memset(pmd->remote_disc_string, 0, sizeof(pmd->remote_disc_string));
37709c7e63aSJakub Grajciar 	strlcpy(pmd->remote_disc_string, (char *)d->string,
37809c7e63aSJakub Grajciar 		sizeof(pmd->remote_disc_string));
37909c7e63aSJakub Grajciar 
38043b815d8SJakub Grajciar 	MIF_LOG(INFO, "Disconnect received: %s", pmd->remote_disc_string);
38109c7e63aSJakub Grajciar 
38243b815d8SJakub Grajciar 	memset(pmd->local_disc_string, 0, 96);
383c41a0495SJakub Grajciar 	memif_disconnect(dev);
38409c7e63aSJakub Grajciar 	return 0;
38509c7e63aSJakub Grajciar }
38609c7e63aSJakub Grajciar 
38709c7e63aSJakub Grajciar static int
memif_msg_enq_ack(struct rte_eth_dev * dev)38809c7e63aSJakub Grajciar memif_msg_enq_ack(struct rte_eth_dev *dev)
38909c7e63aSJakub Grajciar {
39009c7e63aSJakub Grajciar 	struct pmd_internals *pmd = dev->data->dev_private;
39109c7e63aSJakub Grajciar 	struct memif_msg_queue_elt *e = memif_msg_enq(pmd->cc);
39209c7e63aSJakub Grajciar 	if (e == NULL)
39309c7e63aSJakub Grajciar 		return -1;
39409c7e63aSJakub Grajciar 
39509c7e63aSJakub Grajciar 	e->msg.type = MEMIF_MSG_TYPE_ACK;
39609c7e63aSJakub Grajciar 
39709c7e63aSJakub Grajciar 	return 0;
39809c7e63aSJakub Grajciar }
39909c7e63aSJakub Grajciar 
40009c7e63aSJakub Grajciar static int
memif_msg_enq_init(struct rte_eth_dev * dev)40109c7e63aSJakub Grajciar memif_msg_enq_init(struct rte_eth_dev *dev)
40209c7e63aSJakub Grajciar {
40309c7e63aSJakub Grajciar 	struct pmd_internals *pmd = dev->data->dev_private;
40409c7e63aSJakub Grajciar 	struct memif_msg_queue_elt *e = memif_msg_enq(pmd->cc);
405b6c0f464SWeiguo Li 	memif_msg_init_t *i;
40609c7e63aSJakub Grajciar 
40709c7e63aSJakub Grajciar 	if (e == NULL)
40809c7e63aSJakub Grajciar 		return -1;
40909c7e63aSJakub Grajciar 	i = &e->msg.init;
41009c7e63aSJakub Grajciar 	e->msg.type = MEMIF_MSG_TYPE_INIT;
41109c7e63aSJakub Grajciar 	i->version = MEMIF_VERSION;
41209c7e63aSJakub Grajciar 	i->id = pmd->id;
41309c7e63aSJakub Grajciar 	i->mode = MEMIF_INTERFACE_MODE_ETHERNET;
41409c7e63aSJakub Grajciar 
41509c7e63aSJakub Grajciar 	strlcpy((char *)i->name, rte_version(), sizeof(i->name));
41609c7e63aSJakub Grajciar 
41709c7e63aSJakub Grajciar 	if (*pmd->secret != '\0')
41809c7e63aSJakub Grajciar 		strlcpy((char *)i->secret, pmd->secret, sizeof(i->secret));
41909c7e63aSJakub Grajciar 
42009c7e63aSJakub Grajciar 	return 0;
42109c7e63aSJakub Grajciar }
42209c7e63aSJakub Grajciar 
42309c7e63aSJakub Grajciar static int
memif_msg_enq_add_region(struct rte_eth_dev * dev,uint8_t idx)42409c7e63aSJakub Grajciar memif_msg_enq_add_region(struct rte_eth_dev *dev, uint8_t idx)
42509c7e63aSJakub Grajciar {
42609c7e63aSJakub Grajciar 	struct pmd_internals *pmd = dev->data->dev_private;
427c41a0495SJakub Grajciar 	struct pmd_process_private *proc_private = dev->process_private;
42809c7e63aSJakub Grajciar 	struct memif_msg_queue_elt *e = memif_msg_enq(pmd->cc);
42909c7e63aSJakub Grajciar 	memif_msg_add_region_t *ar;
430c41a0495SJakub Grajciar 	struct memif_region *mr = proc_private->regions[idx];
43109c7e63aSJakub Grajciar 
43209c7e63aSJakub Grajciar 	if (e == NULL)
43309c7e63aSJakub Grajciar 		return -1;
43409c7e63aSJakub Grajciar 
43509c7e63aSJakub Grajciar 	ar = &e->msg.add_region;
43609c7e63aSJakub Grajciar 	e->msg.type = MEMIF_MSG_TYPE_ADD_REGION;
43709c7e63aSJakub Grajciar 	e->fd = mr->fd;
43809c7e63aSJakub Grajciar 	ar->index = idx;
43909c7e63aSJakub Grajciar 	ar->size = mr->region_size;
44009c7e63aSJakub Grajciar 
44109c7e63aSJakub Grajciar 	return 0;
44209c7e63aSJakub Grajciar }
44309c7e63aSJakub Grajciar 
44409c7e63aSJakub Grajciar static int
memif_msg_enq_add_ring(struct rte_eth_dev * dev,uint8_t idx,memif_ring_type_t type)44509c7e63aSJakub Grajciar memif_msg_enq_add_ring(struct rte_eth_dev *dev, uint8_t idx,
44609c7e63aSJakub Grajciar 		       memif_ring_type_t type)
44709c7e63aSJakub Grajciar {
44809c7e63aSJakub Grajciar 	struct pmd_internals *pmd = dev->data->dev_private;
44909c7e63aSJakub Grajciar 	struct memif_msg_queue_elt *e = memif_msg_enq(pmd->cc);
45009c7e63aSJakub Grajciar 	struct memif_queue *mq;
45109c7e63aSJakub Grajciar 	memif_msg_add_ring_t *ar;
45209c7e63aSJakub Grajciar 
45309c7e63aSJakub Grajciar 	if (e == NULL)
45409c7e63aSJakub Grajciar 		return -1;
45509c7e63aSJakub Grajciar 
45609c7e63aSJakub Grajciar 	ar = &e->msg.add_ring;
457d250589dSStephen Hemminger 	mq = (type == MEMIF_RING_C2S) ? dev->data->tx_queues[idx] :
45809c7e63aSJakub Grajciar 	    dev->data->rx_queues[idx];
45909c7e63aSJakub Grajciar 
46009c7e63aSJakub Grajciar 	e->msg.type = MEMIF_MSG_TYPE_ADD_RING;
461d61138d4SHarman Kalra 	e->fd = rte_intr_fd_get(mq->intr_handle);
46209c7e63aSJakub Grajciar 	ar->index = idx;
46309c7e63aSJakub Grajciar 	ar->offset = mq->ring_offset;
46409c7e63aSJakub Grajciar 	ar->region = mq->region;
46509c7e63aSJakub Grajciar 	ar->log2_ring_size = mq->log2_ring_size;
466d250589dSStephen Hemminger 	ar->flags = (type == MEMIF_RING_C2S) ? MEMIF_MSG_ADD_RING_FLAG_C2S : 0;
46709c7e63aSJakub Grajciar 	ar->private_hdr_size = 0;
46809c7e63aSJakub Grajciar 
46909c7e63aSJakub Grajciar 	return 0;
47009c7e63aSJakub Grajciar }
47109c7e63aSJakub Grajciar 
47209c7e63aSJakub Grajciar static int
memif_msg_enq_connect(struct rte_eth_dev * dev)47309c7e63aSJakub Grajciar memif_msg_enq_connect(struct rte_eth_dev *dev)
47409c7e63aSJakub Grajciar {
47509c7e63aSJakub Grajciar 	struct pmd_internals *pmd = dev->data->dev_private;
47609c7e63aSJakub Grajciar 	struct memif_msg_queue_elt *e = memif_msg_enq(pmd->cc);
47709c7e63aSJakub Grajciar 	memif_msg_connect_t *c;
47809c7e63aSJakub Grajciar 
47909c7e63aSJakub Grajciar 	if (e == NULL)
48009c7e63aSJakub Grajciar 		return -1;
48109c7e63aSJakub Grajciar 
48209c7e63aSJakub Grajciar 	c = &e->msg.connect;
48309c7e63aSJakub Grajciar 	e->msg.type = MEMIF_MSG_TYPE_CONNECT;
48443b815d8SJakub Grajciar 	strlcpy((char *)c->if_name, dev->data->name, sizeof(c->if_name));
48509c7e63aSJakub Grajciar 
48609c7e63aSJakub Grajciar 	return 0;
48709c7e63aSJakub Grajciar }
48809c7e63aSJakub Grajciar 
48909c7e63aSJakub Grajciar static int
memif_msg_enq_connected(struct rte_eth_dev * dev)49009c7e63aSJakub Grajciar memif_msg_enq_connected(struct rte_eth_dev *dev)
49109c7e63aSJakub Grajciar {
49209c7e63aSJakub Grajciar 	struct pmd_internals *pmd = dev->data->dev_private;
49309c7e63aSJakub Grajciar 	struct memif_msg_queue_elt *e = memif_msg_enq(pmd->cc);
49409c7e63aSJakub Grajciar 	memif_msg_connected_t *c;
49509c7e63aSJakub Grajciar 
49609c7e63aSJakub Grajciar 	if (e == NULL)
49709c7e63aSJakub Grajciar 		return -1;
49809c7e63aSJakub Grajciar 
49909c7e63aSJakub Grajciar 	c = &e->msg.connected;
50009c7e63aSJakub Grajciar 	e->msg.type = MEMIF_MSG_TYPE_CONNECTED;
50143b815d8SJakub Grajciar 	strlcpy((char *)c->if_name, dev->data->name, sizeof(c->if_name));
50209c7e63aSJakub Grajciar 
50309c7e63aSJakub Grajciar 	return 0;
50409c7e63aSJakub Grajciar }
50509c7e63aSJakub Grajciar 
50609c7e63aSJakub Grajciar static void
memif_intr_unregister_handler(struct rte_intr_handle * intr_handle,void * arg)50709c7e63aSJakub Grajciar memif_intr_unregister_handler(struct rte_intr_handle *intr_handle, void *arg)
50809c7e63aSJakub Grajciar {
50909c7e63aSJakub Grajciar 	struct memif_msg_queue_elt *elt;
51009c7e63aSJakub Grajciar 	struct memif_control_channel *cc = arg;
51109c7e63aSJakub Grajciar 
51209c7e63aSJakub Grajciar 	/* close control channel fd */
513aedd054cSHarman Kalra 	if (rte_intr_fd_get(intr_handle) >= 0)
514d61138d4SHarman Kalra 		close(rte_intr_fd_get(intr_handle));
51509c7e63aSJakub Grajciar 	/* clear message queue */
51609c7e63aSJakub Grajciar 	while ((elt = TAILQ_FIRST(&cc->msg_queue)) != NULL) {
51709c7e63aSJakub Grajciar 		TAILQ_REMOVE(&cc->msg_queue, elt, next);
51809c7e63aSJakub Grajciar 		rte_free(elt);
51909c7e63aSJakub Grajciar 	}
520d61138d4SHarman Kalra 	rte_intr_instance_free(cc->intr_handle);
52109c7e63aSJakub Grajciar 	/* free control channel */
52209c7e63aSJakub Grajciar 	rte_free(cc);
52309c7e63aSJakub Grajciar }
52409c7e63aSJakub Grajciar 
52509c7e63aSJakub Grajciar void
memif_disconnect(struct rte_eth_dev * dev)52609c7e63aSJakub Grajciar memif_disconnect(struct rte_eth_dev *dev)
52709c7e63aSJakub Grajciar {
52809c7e63aSJakub Grajciar 	struct pmd_internals *pmd = dev->data->dev_private;
52909c7e63aSJakub Grajciar 	struct memif_msg_queue_elt *elt, *next;
53009c7e63aSJakub Grajciar 	struct memif_queue *mq;
53109c7e63aSJakub Grajciar 	struct rte_intr_handle *ih;
53209c7e63aSJakub Grajciar 	int i;
53309c7e63aSJakub Grajciar 	int ret;
53409c7e63aSJakub Grajciar 
535295968d1SFerruh Yigit 	dev->data->dev_link.link_status = RTE_ETH_LINK_DOWN;
536c41a0495SJakub Grajciar 	pmd->flags &= ~ETH_MEMIF_FLAG_CONNECTING;
537c41a0495SJakub Grajciar 	pmd->flags &= ~ETH_MEMIF_FLAG_CONNECTED;
538c41a0495SJakub Grajciar 
539cbbbbd33SJúlius Milan 	rte_spinlock_lock(&pmd->cc_lock);
54009c7e63aSJakub Grajciar 	if (pmd->cc != NULL) {
54109c7e63aSJakub Grajciar 		/* Clear control message queue (except disconnect message if any). */
54209c7e63aSJakub Grajciar 		for (elt = TAILQ_FIRST(&pmd->cc->msg_queue); elt != NULL; elt = next) {
54309c7e63aSJakub Grajciar 			next = TAILQ_NEXT(elt, next);
54409c7e63aSJakub Grajciar 			if (elt->msg.type != MEMIF_MSG_TYPE_DISCONNECT) {
54509c7e63aSJakub Grajciar 				TAILQ_REMOVE(&pmd->cc->msg_queue, elt, next);
54609c7e63aSJakub Grajciar 				rte_free(elt);
54709c7e63aSJakub Grajciar 			}
54809c7e63aSJakub Grajciar 		}
54909c7e63aSJakub Grajciar 		/* send disconnect message (if there is any in queue) */
55009c7e63aSJakub Grajciar 		memif_msg_send_from_queue(pmd->cc);
55109c7e63aSJakub Grajciar 
55209c7e63aSJakub Grajciar 		/* at this point, there should be no more messages in queue */
55309c7e63aSJakub Grajciar 		if (TAILQ_FIRST(&pmd->cc->msg_queue) != NULL) {
55409c7e63aSJakub Grajciar 			MIF_LOG(WARNING,
555c41a0495SJakub Grajciar 				"Unexpected message(s) in message queue.");
55609c7e63aSJakub Grajciar 		}
55709c7e63aSJakub Grajciar 
558d61138d4SHarman Kalra 		ih = pmd->cc->intr_handle;
559d61138d4SHarman Kalra 		if (rte_intr_fd_get(ih) > 0) {
56009c7e63aSJakub Grajciar 			ret = rte_intr_callback_unregister(ih,
56109c7e63aSJakub Grajciar 							memif_intr_handler,
56209c7e63aSJakub Grajciar 							pmd->cc);
56309c7e63aSJakub Grajciar 			/*
56409c7e63aSJakub Grajciar 			 * If callback is active (disconnecting based on
56509c7e63aSJakub Grajciar 			 * received control message).
56609c7e63aSJakub Grajciar 			 */
56709c7e63aSJakub Grajciar 			if (ret == -EAGAIN) {
56809c7e63aSJakub Grajciar 				ret = rte_intr_callback_unregister_pending(ih,
56909c7e63aSJakub Grajciar 							memif_intr_handler,
57009c7e63aSJakub Grajciar 							pmd->cc,
57109c7e63aSJakub Grajciar 							memif_intr_unregister_handler);
57209c7e63aSJakub Grajciar 			} else if (ret > 0) {
573d61138d4SHarman Kalra 				close(rte_intr_fd_get(ih));
574d61138d4SHarman Kalra 				rte_intr_instance_free(ih);
57509c7e63aSJakub Grajciar 				rte_free(pmd->cc);
57609c7e63aSJakub Grajciar 			}
57709c7e63aSJakub Grajciar 			pmd->cc = NULL;
57809c7e63aSJakub Grajciar 			if (ret <= 0)
579c41a0495SJakub Grajciar 				MIF_LOG(WARNING,
580c41a0495SJakub Grajciar 					"Failed to unregister control channel callback.");
58109c7e63aSJakub Grajciar 		}
58209c7e63aSJakub Grajciar 	}
583cbbbbd33SJúlius Milan 	rte_spinlock_unlock(&pmd->cc_lock);
58409c7e63aSJakub Grajciar 
58509c7e63aSJakub Grajciar 	/* unconfig interrupts */
586d250589dSStephen Hemminger 	for (i = 0; i < pmd->cfg.num_c2s_rings; i++) {
587d250589dSStephen Hemminger 		if (pmd->role == MEMIF_ROLE_CLIENT) {
58809c7e63aSJakub Grajciar 			if (dev->data->tx_queues != NULL)
58909c7e63aSJakub Grajciar 				mq = dev->data->tx_queues[i];
59009c7e63aSJakub Grajciar 			else
59109c7e63aSJakub Grajciar 				continue;
59209c7e63aSJakub Grajciar 		} else {
59309c7e63aSJakub Grajciar 			if (dev->data->rx_queues != NULL)
59409c7e63aSJakub Grajciar 				mq = dev->data->rx_queues[i];
59509c7e63aSJakub Grajciar 			else
59609c7e63aSJakub Grajciar 				continue;
59709c7e63aSJakub Grajciar 		}
598d61138d4SHarman Kalra 
599d61138d4SHarman Kalra 		if (rte_intr_fd_get(mq->intr_handle) > 0) {
600d61138d4SHarman Kalra 			close(rte_intr_fd_get(mq->intr_handle));
601d61138d4SHarman Kalra 			rte_intr_fd_set(mq->intr_handle, -1);
60209c7e63aSJakub Grajciar 		}
60309c7e63aSJakub Grajciar 	}
604d250589dSStephen Hemminger 	for (i = 0; i < pmd->cfg.num_s2c_rings; i++) {
605d250589dSStephen Hemminger 		if (pmd->role == MEMIF_ROLE_SERVER) {
60609c7e63aSJakub Grajciar 			if (dev->data->tx_queues != NULL)
60709c7e63aSJakub Grajciar 				mq = dev->data->tx_queues[i];
60809c7e63aSJakub Grajciar 			else
60909c7e63aSJakub Grajciar 				continue;
61009c7e63aSJakub Grajciar 		} else {
61109c7e63aSJakub Grajciar 			if (dev->data->rx_queues != NULL)
61209c7e63aSJakub Grajciar 				mq = dev->data->rx_queues[i];
61309c7e63aSJakub Grajciar 			else
61409c7e63aSJakub Grajciar 				continue;
61509c7e63aSJakub Grajciar 		}
616d61138d4SHarman Kalra 
617d61138d4SHarman Kalra 		if (rte_intr_fd_get(mq->intr_handle) > 0) {
618d61138d4SHarman Kalra 			close(rte_intr_fd_get(mq->intr_handle));
619d61138d4SHarman Kalra 			rte_intr_fd_set(mq->intr_handle, -1);
62009c7e63aSJakub Grajciar 		}
62109c7e63aSJakub Grajciar 	}
62209c7e63aSJakub Grajciar 
62343b815d8SJakub Grajciar 	memif_free_regions(dev);
62409c7e63aSJakub Grajciar 
62509c7e63aSJakub Grajciar 	/* reset connection configuration */
62609c7e63aSJakub Grajciar 	memset(&pmd->run, 0, sizeof(pmd->run));
62709c7e63aSJakub Grajciar 
628cbbbbd33SJúlius Milan 	MIF_LOG(DEBUG, "Disconnected, id: %d, role: %s.", pmd->id,
629d250589dSStephen Hemminger 		(pmd->role == MEMIF_ROLE_SERVER) ? "server" : "client");
63009c7e63aSJakub Grajciar }
63109c7e63aSJakub Grajciar 
63209c7e63aSJakub Grajciar static int
memif_msg_receive(struct memif_control_channel * cc)63309c7e63aSJakub Grajciar memif_msg_receive(struct memif_control_channel *cc)
63409c7e63aSJakub Grajciar {
63509c7e63aSJakub Grajciar 	char ctl[CMSG_SPACE(sizeof(int)) +
63609c7e63aSJakub Grajciar 		 CMSG_SPACE(sizeof(struct ucred))] = { 0 };
63709c7e63aSJakub Grajciar 	struct msghdr mh = { 0 };
63809c7e63aSJakub Grajciar 	struct iovec iov[1];
63909c7e63aSJakub Grajciar 	memif_msg_t msg = { 0 };
64009c7e63aSJakub Grajciar 	ssize_t size;
64109c7e63aSJakub Grajciar 	int ret = 0;
64209c7e63aSJakub Grajciar 	struct ucred *cr __rte_unused;
64309c7e63aSJakub Grajciar 	cr = 0;
64409c7e63aSJakub Grajciar 	struct cmsghdr *cmsg;
64509c7e63aSJakub Grajciar 	int afd = -1;
64609c7e63aSJakub Grajciar 	int i;
64709c7e63aSJakub Grajciar 	struct pmd_internals *pmd;
648c41a0495SJakub Grajciar 	struct pmd_process_private *proc_private;
64909c7e63aSJakub Grajciar 
65009c7e63aSJakub Grajciar 	iov[0].iov_base = (void *)&msg;
65109c7e63aSJakub Grajciar 	iov[0].iov_len = sizeof(memif_msg_t);
65209c7e63aSJakub Grajciar 	mh.msg_iov = iov;
65309c7e63aSJakub Grajciar 	mh.msg_iovlen = 1;
65409c7e63aSJakub Grajciar 	mh.msg_control = ctl;
65509c7e63aSJakub Grajciar 	mh.msg_controllen = sizeof(ctl);
65609c7e63aSJakub Grajciar 
657aedd054cSHarman Kalra 	if (rte_intr_fd_get(cc->intr_handle) < 0)
658aedd054cSHarman Kalra 		return -1;
659aedd054cSHarman Kalra 
660d61138d4SHarman Kalra 	size = recvmsg(rte_intr_fd_get(cc->intr_handle), &mh, 0);
66109c7e63aSJakub Grajciar 	if (size != sizeof(memif_msg_t)) {
662cbbbbd33SJúlius Milan 		MIF_LOG(DEBUG, "Invalid message size = %zd", size);
663cbbbbd33SJúlius Milan 		if (size > 0)
664cbbbbd33SJúlius Milan 			/* 0 means end-of-file, negative size means error,
665cbbbbd33SJúlius Milan 			 * don't send further disconnect message in such cases.
666cbbbbd33SJúlius Milan 			 */
66709c7e63aSJakub Grajciar 			memif_msg_enq_disconnect(cc, "Invalid message size", 0);
66809c7e63aSJakub Grajciar 		return -1;
66909c7e63aSJakub Grajciar 	}
67009c7e63aSJakub Grajciar 	MIF_LOG(DEBUG, "Received msg type: %u.", msg.type);
67109c7e63aSJakub Grajciar 
67209c7e63aSJakub Grajciar 	cmsg = CMSG_FIRSTHDR(&mh);
67309c7e63aSJakub Grajciar 	while (cmsg) {
67409c7e63aSJakub Grajciar 		if (cmsg->cmsg_level == SOL_SOCKET) {
67509c7e63aSJakub Grajciar 			if (cmsg->cmsg_type == SCM_CREDENTIALS)
67609c7e63aSJakub Grajciar 				cr = (struct ucred *)CMSG_DATA(cmsg);
67709c7e63aSJakub Grajciar 			else if (cmsg->cmsg_type == SCM_RIGHTS)
67843b815d8SJakub Grajciar 				rte_memcpy(&afd, CMSG_DATA(cmsg), sizeof(int));
67909c7e63aSJakub Grajciar 		}
68009c7e63aSJakub Grajciar 		cmsg = CMSG_NXTHDR(&mh, cmsg);
68109c7e63aSJakub Grajciar 	}
68209c7e63aSJakub Grajciar 
68309c7e63aSJakub Grajciar 	if (cc->dev == NULL && msg.type != MEMIF_MSG_TYPE_INIT) {
68409c7e63aSJakub Grajciar 		MIF_LOG(DEBUG, "Unexpected message.");
68509c7e63aSJakub Grajciar 		memif_msg_enq_disconnect(cc, "Unexpected message", 0);
68609c7e63aSJakub Grajciar 		return -1;
68709c7e63aSJakub Grajciar 	}
68809c7e63aSJakub Grajciar 
68909c7e63aSJakub Grajciar 	/* get device from hash data */
69009c7e63aSJakub Grajciar 	switch (msg.type) {
69109c7e63aSJakub Grajciar 	case MEMIF_MSG_TYPE_ACK:
69209c7e63aSJakub Grajciar 		break;
69309c7e63aSJakub Grajciar 	case MEMIF_MSG_TYPE_HELLO:
69409c7e63aSJakub Grajciar 		ret = memif_msg_receive_hello(cc->dev, &msg);
69509c7e63aSJakub Grajciar 		if (ret < 0)
69609c7e63aSJakub Grajciar 			goto exit;
69709c7e63aSJakub Grajciar 		ret = memif_init_regions_and_queues(cc->dev);
69809c7e63aSJakub Grajciar 		if (ret < 0)
69909c7e63aSJakub Grajciar 			goto exit;
70009c7e63aSJakub Grajciar 		ret = memif_msg_enq_init(cc->dev);
70109c7e63aSJakub Grajciar 		if (ret < 0)
70209c7e63aSJakub Grajciar 			goto exit;
70309c7e63aSJakub Grajciar 		pmd = cc->dev->data->dev_private;
704c41a0495SJakub Grajciar 		proc_private = cc->dev->process_private;
705c41a0495SJakub Grajciar 		for (i = 0; i < proc_private->regions_num; i++) {
70609c7e63aSJakub Grajciar 			ret = memif_msg_enq_add_region(cc->dev, i);
70709c7e63aSJakub Grajciar 			if (ret < 0)
70809c7e63aSJakub Grajciar 				goto exit;
70909c7e63aSJakub Grajciar 		}
710d250589dSStephen Hemminger 		for (i = 0; i < pmd->run.num_c2s_rings; i++) {
71109c7e63aSJakub Grajciar 			ret = memif_msg_enq_add_ring(cc->dev, i,
712d250589dSStephen Hemminger 						     MEMIF_RING_C2S);
71309c7e63aSJakub Grajciar 			if (ret < 0)
71409c7e63aSJakub Grajciar 				goto exit;
71509c7e63aSJakub Grajciar 		}
716d250589dSStephen Hemminger 		for (i = 0; i < pmd->run.num_s2c_rings; i++) {
71709c7e63aSJakub Grajciar 			ret = memif_msg_enq_add_ring(cc->dev, i,
718d250589dSStephen Hemminger 						     MEMIF_RING_S2C);
71909c7e63aSJakub Grajciar 			if (ret < 0)
72009c7e63aSJakub Grajciar 				goto exit;
72109c7e63aSJakub Grajciar 		}
72209c7e63aSJakub Grajciar 		ret = memif_msg_enq_connect(cc->dev);
72309c7e63aSJakub Grajciar 		if (ret < 0)
72409c7e63aSJakub Grajciar 			goto exit;
72509c7e63aSJakub Grajciar 		break;
72609c7e63aSJakub Grajciar 	case MEMIF_MSG_TYPE_INIT:
72709c7e63aSJakub Grajciar 		/*
7287be78d02SJosh Soref 		 * This cc does not have an interface associated with it.
72909c7e63aSJakub Grajciar 		 * If suitable interface is found it will be assigned here.
73009c7e63aSJakub Grajciar 		 */
73109c7e63aSJakub Grajciar 		ret = memif_msg_receive_init(cc, &msg);
73209c7e63aSJakub Grajciar 		if (ret < 0)
73309c7e63aSJakub Grajciar 			goto exit;
73409c7e63aSJakub Grajciar 		ret = memif_msg_enq_ack(cc->dev);
73509c7e63aSJakub Grajciar 		if (ret < 0)
73609c7e63aSJakub Grajciar 			goto exit;
73709c7e63aSJakub Grajciar 		break;
73809c7e63aSJakub Grajciar 	case MEMIF_MSG_TYPE_ADD_REGION:
73909c7e63aSJakub Grajciar 		ret = memif_msg_receive_add_region(cc->dev, &msg, afd);
74009c7e63aSJakub Grajciar 		if (ret < 0)
74109c7e63aSJakub Grajciar 			goto exit;
74209c7e63aSJakub Grajciar 		ret = memif_msg_enq_ack(cc->dev);
74309c7e63aSJakub Grajciar 		if (ret < 0)
74409c7e63aSJakub Grajciar 			goto exit;
74509c7e63aSJakub Grajciar 		break;
74609c7e63aSJakub Grajciar 	case MEMIF_MSG_TYPE_ADD_RING:
74709c7e63aSJakub Grajciar 		ret = memif_msg_receive_add_ring(cc->dev, &msg, afd);
74809c7e63aSJakub Grajciar 		if (ret < 0)
74909c7e63aSJakub Grajciar 			goto exit;
75009c7e63aSJakub Grajciar 		ret = memif_msg_enq_ack(cc->dev);
75109c7e63aSJakub Grajciar 		if (ret < 0)
75209c7e63aSJakub Grajciar 			goto exit;
75309c7e63aSJakub Grajciar 		break;
75409c7e63aSJakub Grajciar 	case MEMIF_MSG_TYPE_CONNECT:
75509c7e63aSJakub Grajciar 		ret = memif_msg_receive_connect(cc->dev, &msg);
75609c7e63aSJakub Grajciar 		if (ret < 0)
75709c7e63aSJakub Grajciar 			goto exit;
75809c7e63aSJakub Grajciar 		ret = memif_msg_enq_connected(cc->dev);
75909c7e63aSJakub Grajciar 		if (ret < 0)
76009c7e63aSJakub Grajciar 			goto exit;
76109c7e63aSJakub Grajciar 		break;
76209c7e63aSJakub Grajciar 	case MEMIF_MSG_TYPE_CONNECTED:
76309c7e63aSJakub Grajciar 		ret = memif_msg_receive_connected(cc->dev, &msg);
76409c7e63aSJakub Grajciar 		break;
76509c7e63aSJakub Grajciar 	case MEMIF_MSG_TYPE_DISCONNECT:
76609c7e63aSJakub Grajciar 		ret = memif_msg_receive_disconnect(cc->dev, &msg);
76709c7e63aSJakub Grajciar 		if (ret < 0)
76809c7e63aSJakub Grajciar 			goto exit;
76909c7e63aSJakub Grajciar 		break;
77009c7e63aSJakub Grajciar 	default:
77109c7e63aSJakub Grajciar 		memif_msg_enq_disconnect(cc, "Unknown message type", 0);
77209c7e63aSJakub Grajciar 		ret = -1;
77309c7e63aSJakub Grajciar 		goto exit;
77409c7e63aSJakub Grajciar 	}
77509c7e63aSJakub Grajciar 
77609c7e63aSJakub Grajciar  exit:
77709c7e63aSJakub Grajciar 	return ret;
77809c7e63aSJakub Grajciar }
77909c7e63aSJakub Grajciar 
78009c7e63aSJakub Grajciar static void
memif_intr_handler(void * arg)78109c7e63aSJakub Grajciar memif_intr_handler(void *arg)
78209c7e63aSJakub Grajciar {
78309c7e63aSJakub Grajciar 	struct memif_control_channel *cc = arg;
78409c7e63aSJakub Grajciar 	int ret;
78509c7e63aSJakub Grajciar 
78609c7e63aSJakub Grajciar 	ret = memif_msg_receive(cc);
78709c7e63aSJakub Grajciar 	/* if driver failed to assign device */
78809c7e63aSJakub Grajciar 	if (cc->dev == NULL) {
78906123fddSJúlius Milan 		memif_msg_send_from_queue(cc);
790d61138d4SHarman Kalra 		ret = rte_intr_callback_unregister_pending(cc->intr_handle,
79109c7e63aSJakub Grajciar 							   memif_intr_handler,
79209c7e63aSJakub Grajciar 							   cc,
79309c7e63aSJakub Grajciar 							   memif_intr_unregister_handler);
79409c7e63aSJakub Grajciar 		if (ret < 0)
79509c7e63aSJakub Grajciar 			MIF_LOG(WARNING,
79609c7e63aSJakub Grajciar 				"Failed to unregister control channel callback.");
79709c7e63aSJakub Grajciar 		return;
79809c7e63aSJakub Grajciar 	}
79909c7e63aSJakub Grajciar 	/* if memif_msg_receive failed */
80009c7e63aSJakub Grajciar 	if (ret < 0)
80109c7e63aSJakub Grajciar 		goto disconnect;
80209c7e63aSJakub Grajciar 
80309c7e63aSJakub Grajciar 	ret = memif_msg_send_from_queue(cc);
80409c7e63aSJakub Grajciar 	if (ret < 0)
80509c7e63aSJakub Grajciar 		goto disconnect;
80609c7e63aSJakub Grajciar 
80709c7e63aSJakub Grajciar 	return;
80809c7e63aSJakub Grajciar 
80909c7e63aSJakub Grajciar  disconnect:
81009c7e63aSJakub Grajciar 	if (cc->dev == NULL) {
81109c7e63aSJakub Grajciar 		MIF_LOG(WARNING, "eth dev not allocated");
81209c7e63aSJakub Grajciar 		return;
81309c7e63aSJakub Grajciar 	}
81409c7e63aSJakub Grajciar 	memif_disconnect(cc->dev);
81509c7e63aSJakub Grajciar }
81609c7e63aSJakub Grajciar 
81709c7e63aSJakub Grajciar static void
memif_listener_handler(void * arg)81809c7e63aSJakub Grajciar memif_listener_handler(void *arg)
81909c7e63aSJakub Grajciar {
82009c7e63aSJakub Grajciar 	struct memif_socket *socket = arg;
82109c7e63aSJakub Grajciar 	int sockfd;
82209c7e63aSJakub Grajciar 	int addr_len;
82309c7e63aSJakub Grajciar 	struct sockaddr_un client;
82409c7e63aSJakub Grajciar 	struct memif_control_channel *cc;
82509c7e63aSJakub Grajciar 	int ret;
82609c7e63aSJakub Grajciar 
82709c7e63aSJakub Grajciar 	addr_len = sizeof(client);
828d61138d4SHarman Kalra 	sockfd = accept(rte_intr_fd_get(socket->intr_handle),
829d61138d4SHarman Kalra 			(struct sockaddr *)&client, (socklen_t *)&addr_len);
83009c7e63aSJakub Grajciar 	if (sockfd < 0) {
83109c7e63aSJakub Grajciar 		MIF_LOG(ERR,
83209c7e63aSJakub Grajciar 			"Failed to accept connection request on socket fd %d",
833d61138d4SHarman Kalra 			rte_intr_fd_get(socket->intr_handle));
83409c7e63aSJakub Grajciar 		return;
83509c7e63aSJakub Grajciar 	}
83609c7e63aSJakub Grajciar 
83709c7e63aSJakub Grajciar 	MIF_LOG(DEBUG, "%s: Connection request accepted.", socket->filename);
83809c7e63aSJakub Grajciar 
83909c7e63aSJakub Grajciar 	cc = rte_zmalloc("memif-cc", sizeof(struct memif_control_channel), 0);
84009c7e63aSJakub Grajciar 	if (cc == NULL) {
84109c7e63aSJakub Grajciar 		MIF_LOG(ERR, "Failed to allocate control channel.");
84209c7e63aSJakub Grajciar 		goto error;
84309c7e63aSJakub Grajciar 	}
84409c7e63aSJakub Grajciar 
845d61138d4SHarman Kalra 	/* Allocate interrupt instance */
846d61138d4SHarman Kalra 	cc->intr_handle = rte_intr_instance_alloc(RTE_INTR_INSTANCE_F_SHARED);
847d61138d4SHarman Kalra 	if (cc->intr_handle == NULL) {
848d61138d4SHarman Kalra 		MIF_LOG(ERR, "Failed to allocate intr handle");
849d61138d4SHarman Kalra 		goto error;
850d61138d4SHarman Kalra 	}
851d61138d4SHarman Kalra 
852d61138d4SHarman Kalra 	if (rte_intr_fd_set(cc->intr_handle, sockfd))
853d61138d4SHarman Kalra 		goto error;
854d61138d4SHarman Kalra 
855d61138d4SHarman Kalra 	if (rte_intr_type_set(cc->intr_handle, RTE_INTR_HANDLE_EXT))
856d61138d4SHarman Kalra 		goto error;
857d61138d4SHarman Kalra 
85809c7e63aSJakub Grajciar 	cc->socket = socket;
85909c7e63aSJakub Grajciar 	cc->dev = NULL;
86009c7e63aSJakub Grajciar 	TAILQ_INIT(&cc->msg_queue);
86109c7e63aSJakub Grajciar 
862d61138d4SHarman Kalra 	ret = rte_intr_callback_register(cc->intr_handle, memif_intr_handler,
863d61138d4SHarman Kalra 					 cc);
86409c7e63aSJakub Grajciar 	if (ret < 0) {
86509c7e63aSJakub Grajciar 		MIF_LOG(ERR, "Failed to register control channel callback.");
86609c7e63aSJakub Grajciar 		goto error;
86709c7e63aSJakub Grajciar 	}
86809c7e63aSJakub Grajciar 
86909c7e63aSJakub Grajciar 	ret = memif_msg_enq_hello(cc);
87009c7e63aSJakub Grajciar 	if (ret < 0) {
87109c7e63aSJakub Grajciar 		MIF_LOG(ERR, "Failed to enqueue hello message.");
87209c7e63aSJakub Grajciar 		goto error;
87309c7e63aSJakub Grajciar 	}
87409c7e63aSJakub Grajciar 	ret = memif_msg_send_from_queue(cc);
87509c7e63aSJakub Grajciar 	if (ret < 0)
87609c7e63aSJakub Grajciar 		goto error;
87709c7e63aSJakub Grajciar 
87809c7e63aSJakub Grajciar 	return;
87909c7e63aSJakub Grajciar 
88009c7e63aSJakub Grajciar  error:
88110bd0ad5SJakub Grajciar 	if (sockfd >= 0) {
88209c7e63aSJakub Grajciar 		close(sockfd);
88309c7e63aSJakub Grajciar 		sockfd = -1;
88409c7e63aSJakub Grajciar 	}
885d61138d4SHarman Kalra 	if (cc != NULL) {
886d61138d4SHarman Kalra 		rte_intr_instance_free(cc->intr_handle);
88709c7e63aSJakub Grajciar 		rte_free(cc);
88809c7e63aSJakub Grajciar 	}
889d61138d4SHarman Kalra }
89009c7e63aSJakub Grajciar 
89109c7e63aSJakub Grajciar static struct memif_socket *
memif_socket_create(char * key,uint8_t listener,bool is_abstract,uid_t owner_uid,gid_t owner_gid)892*f62d5262SJunxiao Shi memif_socket_create(char *key, uint8_t listener, bool is_abstract, uid_t owner_uid, gid_t owner_gid)
89309c7e63aSJakub Grajciar {
89409c7e63aSJakub Grajciar 	struct memif_socket *sock;
8952f865ed0SJakub Grajciar 	struct sockaddr_un un = { 0 };
8964e30586dSNathan Skrzypczak 	uint32_t sunlen;
89709c7e63aSJakub Grajciar 	int sockfd;
89809c7e63aSJakub Grajciar 	int ret;
89909c7e63aSJakub Grajciar 	int on = 1;
90009c7e63aSJakub Grajciar 
90109c7e63aSJakub Grajciar 	sock = rte_zmalloc("memif-socket", sizeof(struct memif_socket), 0);
90209c7e63aSJakub Grajciar 	if (sock == NULL) {
90309c7e63aSJakub Grajciar 		MIF_LOG(ERR, "Failed to allocate memory for memif socket");
90409c7e63aSJakub Grajciar 		return NULL;
90509c7e63aSJakub Grajciar 	}
90609c7e63aSJakub Grajciar 
90709c7e63aSJakub Grajciar 	sock->listener = listener;
908d803feecSJakub Grajciar 	strlcpy(sock->filename, key, MEMIF_SOCKET_UN_SIZE);
90909c7e63aSJakub Grajciar 	TAILQ_INIT(&sock->dev_queue);
91009c7e63aSJakub Grajciar 
91109c7e63aSJakub Grajciar 	if (listener != 0) {
91209c7e63aSJakub Grajciar 		sockfd = socket(AF_UNIX, SOCK_SEQPACKET, 0);
91309c7e63aSJakub Grajciar 		if (sockfd < 0)
91409c7e63aSJakub Grajciar 			goto error;
91509c7e63aSJakub Grajciar 
916d803feecSJakub Grajciar 		un.sun_family = AF_UNIX;
9172f865ed0SJakub Grajciar 		if (is_abstract) {
9182f865ed0SJakub Grajciar 			/* abstract address */
9192f865ed0SJakub Grajciar 			un.sun_path[0] = '\0';
9202f865ed0SJakub Grajciar 			strlcpy(un.sun_path + 1, sock->filename, MEMIF_SOCKET_UN_SIZE - 1);
9214e30586dSNathan Skrzypczak 			sunlen = RTE_MIN(1 + strlen(sock->filename),
9224e30586dSNathan Skrzypczak 					 MEMIF_SOCKET_UN_SIZE) +
9234e30586dSNathan Skrzypczak 				 sizeof(un) - sizeof(un.sun_path);
9242f865ed0SJakub Grajciar 		} else {
9254e30586dSNathan Skrzypczak 			sunlen = sizeof(un);
926d803feecSJakub Grajciar 			strlcpy(un.sun_path, sock->filename, MEMIF_SOCKET_UN_SIZE);
9272f865ed0SJakub Grajciar 		}
92809c7e63aSJakub Grajciar 
92909c7e63aSJakub Grajciar 		ret = setsockopt(sockfd, SOL_SOCKET, SO_PASSCRED, &on,
93009c7e63aSJakub Grajciar 				 sizeof(on));
93109c7e63aSJakub Grajciar 		if (ret < 0)
93209c7e63aSJakub Grajciar 			goto error;
933d803feecSJakub Grajciar 
9344e30586dSNathan Skrzypczak 		ret = bind(sockfd, (struct sockaddr *)&un, sunlen);
93509c7e63aSJakub Grajciar 		if (ret < 0)
93609c7e63aSJakub Grajciar 			goto error;
937d803feecSJakub Grajciar 
93809c7e63aSJakub Grajciar 		ret = listen(sockfd, 1);
93909c7e63aSJakub Grajciar 		if (ret < 0)
94009c7e63aSJakub Grajciar 			goto error;
94109c7e63aSJakub Grajciar 
94243b815d8SJakub Grajciar 		MIF_LOG(DEBUG, "Memif listener socket %s created.", sock->filename);
94309c7e63aSJakub Grajciar 
944*f62d5262SJunxiao Shi 		if (!is_abstract && (owner_uid != (uid_t)-1 || owner_gid != (gid_t)-1)) {
945*f62d5262SJunxiao Shi 			ret = chown(sock->filename, owner_uid, owner_gid);
946*f62d5262SJunxiao Shi 			if (ret < 0) {
947*f62d5262SJunxiao Shi 				MIF_LOG(ERR, "Failed to change listener socket owner");
948*f62d5262SJunxiao Shi 				goto error;
949*f62d5262SJunxiao Shi 			}
950*f62d5262SJunxiao Shi 		}
951*f62d5262SJunxiao Shi 
952d61138d4SHarman Kalra 		/* Allocate interrupt instance */
953d61138d4SHarman Kalra 		sock->intr_handle =
954d61138d4SHarman Kalra 			rte_intr_instance_alloc(RTE_INTR_INSTANCE_F_SHARED);
955d61138d4SHarman Kalra 		if (sock->intr_handle == NULL) {
956d61138d4SHarman Kalra 			MIF_LOG(ERR, "Failed to allocate intr handle");
957d61138d4SHarman Kalra 			goto error;
958d61138d4SHarman Kalra 		}
959d61138d4SHarman Kalra 
960d61138d4SHarman Kalra 		if (rte_intr_fd_set(sock->intr_handle, sockfd))
961d61138d4SHarman Kalra 			goto error;
962d61138d4SHarman Kalra 
963d61138d4SHarman Kalra 		if (rte_intr_type_set(sock->intr_handle, RTE_INTR_HANDLE_EXT))
964d61138d4SHarman Kalra 			goto error;
965d61138d4SHarman Kalra 
966d61138d4SHarman Kalra 		ret = rte_intr_callback_register(sock->intr_handle,
96709c7e63aSJakub Grajciar 						 memif_listener_handler, sock);
96809c7e63aSJakub Grajciar 		if (ret < 0) {
96943b815d8SJakub Grajciar 			MIF_LOG(ERR, "Failed to register interrupt "
97043b815d8SJakub Grajciar 				"callback for listener socket");
97109c7e63aSJakub Grajciar 			return NULL;
97209c7e63aSJakub Grajciar 		}
97309c7e63aSJakub Grajciar 	}
97409c7e63aSJakub Grajciar 
97509c7e63aSJakub Grajciar 	return sock;
97609c7e63aSJakub Grajciar 
97709c7e63aSJakub Grajciar  error:
97843b815d8SJakub Grajciar 	MIF_LOG(ERR, "Failed to setup socket %s: %s", key, strerror(errno));
979d61138d4SHarman Kalra 	if (sock != NULL) {
980d61138d4SHarman Kalra 		rte_intr_instance_free(sock->intr_handle);
98109c7e63aSJakub Grajciar 		rte_free(sock);
982d61138d4SHarman Kalra 	}
98310bd0ad5SJakub Grajciar 	if (sockfd >= 0)
98410bd0ad5SJakub Grajciar 		close(sockfd);
98509c7e63aSJakub Grajciar 	return NULL;
98609c7e63aSJakub Grajciar }
98709c7e63aSJakub Grajciar 
98809c7e63aSJakub Grajciar static struct rte_hash *
memif_create_socket_hash(void)98909c7e63aSJakub Grajciar memif_create_socket_hash(void)
99009c7e63aSJakub Grajciar {
99109c7e63aSJakub Grajciar 	struct rte_hash_parameters params = { 0 };
992b923866cSStephen Hemminger 
99309c7e63aSJakub Grajciar 	params.name = MEMIF_SOCKET_HASH_NAME;
99409c7e63aSJakub Grajciar 	params.entries = 256;
995d803feecSJakub Grajciar 	params.key_len = MEMIF_SOCKET_UN_SIZE;
99609c7e63aSJakub Grajciar 	params.hash_func = rte_jhash;
99709c7e63aSJakub Grajciar 	params.hash_func_init_val = 0;
99818627cedSJunxiao Shi 	params.socket_id = SOCKET_ID_ANY;
99909c7e63aSJakub Grajciar 	return rte_hash_create(&params);
100009c7e63aSJakub Grajciar }
100109c7e63aSJakub Grajciar 
100209c7e63aSJakub Grajciar int
memif_socket_init(struct rte_eth_dev * dev,const char * socket_filename)100309c7e63aSJakub Grajciar memif_socket_init(struct rte_eth_dev *dev, const char *socket_filename)
100409c7e63aSJakub Grajciar {
100509c7e63aSJakub Grajciar 	struct pmd_internals *pmd = dev->data->dev_private;
100609c7e63aSJakub Grajciar 	struct memif_socket *socket = NULL;
100709c7e63aSJakub Grajciar 	struct memif_socket_dev_list_elt *elt;
100809c7e63aSJakub Grajciar 	struct pmd_internals *tmp_pmd;
100909c7e63aSJakub Grajciar 	struct rte_hash *hash;
101009c7e63aSJakub Grajciar 	int ret;
1011d803feecSJakub Grajciar 	char key[MEMIF_SOCKET_UN_SIZE];
101209c7e63aSJakub Grajciar 
101309c7e63aSJakub Grajciar 	hash = rte_hash_find_existing(MEMIF_SOCKET_HASH_NAME);
101409c7e63aSJakub Grajciar 	if (hash == NULL) {
101509c7e63aSJakub Grajciar 		hash = memif_create_socket_hash();
101609c7e63aSJakub Grajciar 		if (hash == NULL) {
101709c7e63aSJakub Grajciar 			MIF_LOG(ERR, "Failed to create memif socket hash.");
101809c7e63aSJakub Grajciar 			return -1;
101909c7e63aSJakub Grajciar 		}
102009c7e63aSJakub Grajciar 	}
102109c7e63aSJakub Grajciar 
1022d803feecSJakub Grajciar 	memset(key, 0, MEMIF_SOCKET_UN_SIZE);
1023d803feecSJakub Grajciar 	strlcpy(key, socket_filename, MEMIF_SOCKET_UN_SIZE);
102409c7e63aSJakub Grajciar 	ret = rte_hash_lookup_data(hash, key, (void **)&socket);
102509c7e63aSJakub Grajciar 	if (ret < 0) {
102643b815d8SJakub Grajciar 		socket = memif_socket_create(key,
1027d250589dSStephen Hemminger 			(pmd->role == MEMIF_ROLE_CLIENT) ? 0 : 1,
1028*f62d5262SJunxiao Shi 			pmd->flags & ETH_MEMIF_FLAG_SOCKET_ABSTRACT,
1029*f62d5262SJunxiao Shi 			pmd->owner_uid, pmd->owner_gid);
103009c7e63aSJakub Grajciar 		if (socket == NULL)
103109c7e63aSJakub Grajciar 			return -1;
103209c7e63aSJakub Grajciar 		ret = rte_hash_add_key_data(hash, key, socket);
103309c7e63aSJakub Grajciar 		if (ret < 0) {
103409c7e63aSJakub Grajciar 			MIF_LOG(ERR, "Failed to add socket to socket hash.");
103509c7e63aSJakub Grajciar 			return ret;
103609c7e63aSJakub Grajciar 		}
103709c7e63aSJakub Grajciar 	}
103809c7e63aSJakub Grajciar 	pmd->socket_filename = socket->filename;
103909c7e63aSJakub Grajciar 
104009c7e63aSJakub Grajciar 	TAILQ_FOREACH(elt, &socket->dev_queue, next) {
104109c7e63aSJakub Grajciar 		tmp_pmd = elt->dev->data->dev_private;
1042cbbbbd33SJúlius Milan 		if (tmp_pmd->id == pmd->id && tmp_pmd->role == pmd->role) {
1043cbbbbd33SJúlius Milan 			MIF_LOG(ERR, "Two interfaces with the same id (%d) can "
1044cbbbbd33SJúlius Milan 				"not have the same role.", pmd->id);
104509c7e63aSJakub Grajciar 			return -1;
104609c7e63aSJakub Grajciar 		}
104709c7e63aSJakub Grajciar 	}
104809c7e63aSJakub Grajciar 
104909c7e63aSJakub Grajciar 	elt = rte_malloc("pmd-queue", sizeof(struct memif_socket_dev_list_elt), 0);
105009c7e63aSJakub Grajciar 	if (elt == NULL) {
105143b815d8SJakub Grajciar 		MIF_LOG(ERR, "Failed to add device to socket device list.");
105209c7e63aSJakub Grajciar 		return -1;
105309c7e63aSJakub Grajciar 	}
105409c7e63aSJakub Grajciar 	elt->dev = dev;
105509c7e63aSJakub Grajciar 	TAILQ_INSERT_TAIL(&socket->dev_queue, elt, next);
105609c7e63aSJakub Grajciar 
105709c7e63aSJakub Grajciar 	return 0;
105809c7e63aSJakub Grajciar }
105909c7e63aSJakub Grajciar 
106009c7e63aSJakub Grajciar void
memif_socket_remove_device(struct rte_eth_dev * dev)106109c7e63aSJakub Grajciar memif_socket_remove_device(struct rte_eth_dev *dev)
106209c7e63aSJakub Grajciar {
106309c7e63aSJakub Grajciar 	struct pmd_internals *pmd = dev->data->dev_private;
106409c7e63aSJakub Grajciar 	struct memif_socket *socket = NULL;
106509c7e63aSJakub Grajciar 	struct memif_socket_dev_list_elt *elt, *next;
106609c7e63aSJakub Grajciar 	struct rte_hash *hash;
106710bd0ad5SJakub Grajciar 	int ret;
106809c7e63aSJakub Grajciar 
106909c7e63aSJakub Grajciar 	hash = rte_hash_find_existing(MEMIF_SOCKET_HASH_NAME);
107009c7e63aSJakub Grajciar 	if (hash == NULL)
107109c7e63aSJakub Grajciar 		return;
107209c7e63aSJakub Grajciar 
107309c7e63aSJakub Grajciar 	if (pmd->socket_filename == NULL)
107409c7e63aSJakub Grajciar 		return;
107509c7e63aSJakub Grajciar 
107609c7e63aSJakub Grajciar 	if (rte_hash_lookup_data(hash, pmd->socket_filename, (void **)&socket) < 0)
107709c7e63aSJakub Grajciar 		return;
107809c7e63aSJakub Grajciar 
107909c7e63aSJakub Grajciar 	for (elt = TAILQ_FIRST(&socket->dev_queue); elt != NULL; elt = next) {
108009c7e63aSJakub Grajciar 		next = TAILQ_NEXT(elt, next);
108109c7e63aSJakub Grajciar 		if (elt->dev == dev) {
108209c7e63aSJakub Grajciar 			TAILQ_REMOVE(&socket->dev_queue, elt, next);
108309c7e63aSJakub Grajciar 			rte_free(elt);
108409c7e63aSJakub Grajciar 			pmd->socket_filename = NULL;
108509c7e63aSJakub Grajciar 		}
108609c7e63aSJakub Grajciar 	}
108709c7e63aSJakub Grajciar 
108809c7e63aSJakub Grajciar 	/* remove socket, if this was the last device using it */
108909c7e63aSJakub Grajciar 	if (TAILQ_EMPTY(&socket->dev_queue)) {
109009c7e63aSJakub Grajciar 		rte_hash_del_key(hash, socket->filename);
10912f865ed0SJakub Grajciar 		if (socket->listener && !(pmd->flags & ETH_MEMIF_FLAG_SOCKET_ABSTRACT)) {
109209c7e63aSJakub Grajciar 			/* remove listener socket file,
109309c7e63aSJakub Grajciar 			 * so we can create new one later.
109409c7e63aSJakub Grajciar 			 */
109510bd0ad5SJakub Grajciar 			ret = remove(socket->filename);
109610bd0ad5SJakub Grajciar 			if (ret < 0)
109710bd0ad5SJakub Grajciar 				MIF_LOG(ERR, "Failed to remove socket file: %s",
109810bd0ad5SJakub Grajciar 					socket->filename);
109909c7e63aSJakub Grajciar 		}
1100d61138d4SHarman Kalra 		if (pmd->role != MEMIF_ROLE_CLIENT)
1101d61138d4SHarman Kalra 			rte_intr_instance_free(socket->intr_handle);
110209c7e63aSJakub Grajciar 		rte_free(socket);
110309c7e63aSJakub Grajciar 	}
110409c7e63aSJakub Grajciar }
110509c7e63aSJakub Grajciar 
110609c7e63aSJakub Grajciar int
memif_connect_server(struct rte_eth_dev * dev)1107d250589dSStephen Hemminger memif_connect_server(struct rte_eth_dev *dev)
110809c7e63aSJakub Grajciar {
110909c7e63aSJakub Grajciar 	struct pmd_internals *pmd = dev->data->dev_private;
111009c7e63aSJakub Grajciar 
111109c7e63aSJakub Grajciar 	memset(pmd->local_disc_string, 0, ETH_MEMIF_DISC_STRING_SIZE);
111209c7e63aSJakub Grajciar 	memset(pmd->remote_disc_string, 0, ETH_MEMIF_DISC_STRING_SIZE);
111309c7e63aSJakub Grajciar 	pmd->flags &= ~ETH_MEMIF_FLAG_DISABLED;
111409c7e63aSJakub Grajciar 	return 0;
111509c7e63aSJakub Grajciar }
111609c7e63aSJakub Grajciar 
111709c7e63aSJakub Grajciar int
memif_connect_client(struct rte_eth_dev * dev)1118d250589dSStephen Hemminger memif_connect_client(struct rte_eth_dev *dev)
111909c7e63aSJakub Grajciar {
112009c7e63aSJakub Grajciar 	int sockfd;
112109c7e63aSJakub Grajciar 	int ret;
11224e30586dSNathan Skrzypczak 	uint32_t sunlen;
11232f865ed0SJakub Grajciar 	struct sockaddr_un sun = { 0 };
112409c7e63aSJakub Grajciar 	struct pmd_internals *pmd = dev->data->dev_private;
112509c7e63aSJakub Grajciar 
112609c7e63aSJakub Grajciar 	memset(pmd->local_disc_string, 0, ETH_MEMIF_DISC_STRING_SIZE);
112709c7e63aSJakub Grajciar 	memset(pmd->remote_disc_string, 0, ETH_MEMIF_DISC_STRING_SIZE);
112809c7e63aSJakub Grajciar 	pmd->flags &= ~ETH_MEMIF_FLAG_DISABLED;
112909c7e63aSJakub Grajciar 
113009c7e63aSJakub Grajciar 	sockfd = socket(AF_UNIX, SOCK_SEQPACKET, 0);
113109c7e63aSJakub Grajciar 	if (sockfd < 0) {
113243b815d8SJakub Grajciar 		MIF_LOG(ERR, "Failed to open socket.");
113309c7e63aSJakub Grajciar 		return -1;
113409c7e63aSJakub Grajciar 	}
113509c7e63aSJakub Grajciar 
113609c7e63aSJakub Grajciar 	sun.sun_family = AF_UNIX;
11374e30586dSNathan Skrzypczak 	sunlen = sizeof(struct sockaddr_un);
11382f865ed0SJakub Grajciar 	if (pmd->flags & ETH_MEMIF_FLAG_SOCKET_ABSTRACT) {
11392f865ed0SJakub Grajciar 		/* abstract address */
11402f865ed0SJakub Grajciar 		sun.sun_path[0] = '\0';
11412f865ed0SJakub Grajciar 		strlcpy(sun.sun_path + 1,  pmd->socket_filename, MEMIF_SOCKET_UN_SIZE - 1);
11424e30586dSNathan Skrzypczak 		sunlen = RTE_MIN(strlen(pmd->socket_filename) + 1,
11434e30586dSNathan Skrzypczak 				 MEMIF_SOCKET_UN_SIZE) +
11444e30586dSNathan Skrzypczak 			 sizeof(sun) - sizeof(sun.sun_path);
11452f865ed0SJakub Grajciar 	} else {
11462f865ed0SJakub Grajciar 		strlcpy(sun.sun_path,  pmd->socket_filename, MEMIF_SOCKET_UN_SIZE);
11472f865ed0SJakub Grajciar 	}
114809c7e63aSJakub Grajciar 
11494e30586dSNathan Skrzypczak 	ret = connect(sockfd, (struct sockaddr *)&sun, sunlen);
115009c7e63aSJakub Grajciar 	if (ret < 0) {
115143b815d8SJakub Grajciar 		MIF_LOG(ERR, "Failed to connect socket: %s.", pmd->socket_filename);
115209c7e63aSJakub Grajciar 		goto error;
115309c7e63aSJakub Grajciar 	}
115409c7e63aSJakub Grajciar 
115543b815d8SJakub Grajciar 	MIF_LOG(DEBUG, "Memif socket: %s connected.", pmd->socket_filename);
115609c7e63aSJakub Grajciar 
115709c7e63aSJakub Grajciar 	pmd->cc = rte_zmalloc("memif-cc",
115809c7e63aSJakub Grajciar 			      sizeof(struct memif_control_channel), 0);
115909c7e63aSJakub Grajciar 	if (pmd->cc == NULL) {
116043b815d8SJakub Grajciar 		MIF_LOG(ERR, "Failed to allocate control channel.");
116109c7e63aSJakub Grajciar 		goto error;
116209c7e63aSJakub Grajciar 	}
116309c7e63aSJakub Grajciar 
1164d61138d4SHarman Kalra 	/* Allocate interrupt instance */
1165d61138d4SHarman Kalra 	pmd->cc->intr_handle =
1166d61138d4SHarman Kalra 		rte_intr_instance_alloc(RTE_INTR_INSTANCE_F_SHARED);
1167d61138d4SHarman Kalra 	if (pmd->cc->intr_handle == NULL) {
1168d61138d4SHarman Kalra 		MIF_LOG(ERR, "Failed to allocate intr handle");
1169d61138d4SHarman Kalra 		goto error;
1170d61138d4SHarman Kalra 	}
1171d61138d4SHarman Kalra 
1172d61138d4SHarman Kalra 	if (rte_intr_fd_set(pmd->cc->intr_handle, sockfd))
1173d61138d4SHarman Kalra 		goto error;
1174d61138d4SHarman Kalra 
1175d61138d4SHarman Kalra 	if (rte_intr_type_set(pmd->cc->intr_handle, RTE_INTR_HANDLE_EXT))
1176d61138d4SHarman Kalra 		goto error;
1177d61138d4SHarman Kalra 
117809c7e63aSJakub Grajciar 	pmd->cc->socket = NULL;
117909c7e63aSJakub Grajciar 	pmd->cc->dev = dev;
118009c7e63aSJakub Grajciar 	TAILQ_INIT(&pmd->cc->msg_queue);
118109c7e63aSJakub Grajciar 
1182d61138d4SHarman Kalra 	ret = rte_intr_callback_register(pmd->cc->intr_handle,
118309c7e63aSJakub Grajciar 					 memif_intr_handler, pmd->cc);
118409c7e63aSJakub Grajciar 	if (ret < 0) {
118543b815d8SJakub Grajciar 		MIF_LOG(ERR, "Failed to register interrupt callback for control fd");
118609c7e63aSJakub Grajciar 		goto error;
118709c7e63aSJakub Grajciar 	}
118809c7e63aSJakub Grajciar 
118909c7e63aSJakub Grajciar 	return 0;
119009c7e63aSJakub Grajciar 
119109c7e63aSJakub Grajciar  error:
119210bd0ad5SJakub Grajciar 	if (sockfd >= 0) {
119309c7e63aSJakub Grajciar 		close(sockfd);
119409c7e63aSJakub Grajciar 		sockfd = -1;
119509c7e63aSJakub Grajciar 	}
119609c7e63aSJakub Grajciar 	if (pmd->cc != NULL) {
1197d61138d4SHarman Kalra 		rte_intr_instance_free(pmd->cc->intr_handle);
119809c7e63aSJakub Grajciar 		rte_free(pmd->cc);
119909c7e63aSJakub Grajciar 		pmd->cc = NULL;
120009c7e63aSJakub Grajciar 	}
120109c7e63aSJakub Grajciar 	return -1;
120209c7e63aSJakub Grajciar }
1203