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/un.h> 1109c7e63aSJakub Grajciar #include <sys/ioctl.h> 1209c7e63aSJakub Grajciar #include <errno.h> 1309c7e63aSJakub Grajciar 1409c7e63aSJakub Grajciar #include <rte_version.h> 1509c7e63aSJakub Grajciar #include <rte_mbuf.h> 1609c7e63aSJakub Grajciar #include <rte_ether.h> 1709c7e63aSJakub Grajciar #include <rte_ethdev_driver.h> 1809c7e63aSJakub Grajciar #include <rte_ethdev_vdev.h> 1909c7e63aSJakub Grajciar #include <rte_malloc.h> 2009c7e63aSJakub Grajciar #include <rte_kvargs.h> 2109c7e63aSJakub Grajciar #include <rte_bus_vdev.h> 2209c7e63aSJakub Grajciar #include <rte_hash.h> 2309c7e63aSJakub Grajciar #include <rte_jhash.h> 2409c7e63aSJakub Grajciar #include <rte_string_fns.h> 2509c7e63aSJakub Grajciar 2609c7e63aSJakub Grajciar #include "rte_eth_memif.h" 2709c7e63aSJakub Grajciar #include "memif_socket.h" 2809c7e63aSJakub Grajciar 2909c7e63aSJakub Grajciar static void memif_intr_handler(void *arg); 3009c7e63aSJakub Grajciar 3109c7e63aSJakub Grajciar static ssize_t 3209c7e63aSJakub Grajciar memif_msg_send(int fd, memif_msg_t *msg, int afd) 3309c7e63aSJakub Grajciar { 3409c7e63aSJakub Grajciar struct msghdr mh = { 0 }; 3509c7e63aSJakub Grajciar struct iovec iov[1]; 3609c7e63aSJakub Grajciar struct cmsghdr *cmsg; 3709c7e63aSJakub Grajciar char ctl[CMSG_SPACE(sizeof(int))]; 3809c7e63aSJakub Grajciar 3909c7e63aSJakub Grajciar iov[0].iov_base = msg; 4009c7e63aSJakub Grajciar iov[0].iov_len = sizeof(memif_msg_t); 4109c7e63aSJakub Grajciar mh.msg_iov = iov; 4209c7e63aSJakub Grajciar mh.msg_iovlen = 1; 4309c7e63aSJakub Grajciar 4409c7e63aSJakub Grajciar if (afd > 0) { 4509c7e63aSJakub Grajciar memset(&ctl, 0, sizeof(ctl)); 4609c7e63aSJakub Grajciar mh.msg_control = ctl; 4709c7e63aSJakub Grajciar mh.msg_controllen = sizeof(ctl); 4809c7e63aSJakub Grajciar cmsg = CMSG_FIRSTHDR(&mh); 4909c7e63aSJakub Grajciar cmsg->cmsg_len = CMSG_LEN(sizeof(int)); 5009c7e63aSJakub Grajciar cmsg->cmsg_level = SOL_SOCKET; 5109c7e63aSJakub Grajciar cmsg->cmsg_type = SCM_RIGHTS; 5209c7e63aSJakub Grajciar rte_memcpy(CMSG_DATA(cmsg), &afd, sizeof(int)); 5309c7e63aSJakub Grajciar } 5409c7e63aSJakub Grajciar 5509c7e63aSJakub Grajciar return sendmsg(fd, &mh, 0); 5609c7e63aSJakub Grajciar } 5709c7e63aSJakub Grajciar 5809c7e63aSJakub Grajciar static int 5909c7e63aSJakub Grajciar memif_msg_send_from_queue(struct memif_control_channel *cc) 6009c7e63aSJakub Grajciar { 6109c7e63aSJakub Grajciar ssize_t size; 6209c7e63aSJakub Grajciar int ret = 0; 6309c7e63aSJakub Grajciar struct memif_msg_queue_elt *e; 6409c7e63aSJakub Grajciar 6509c7e63aSJakub Grajciar e = TAILQ_FIRST(&cc->msg_queue); 6609c7e63aSJakub Grajciar if (e == NULL) 6709c7e63aSJakub Grajciar return 0; 6809c7e63aSJakub Grajciar 6909c7e63aSJakub Grajciar size = memif_msg_send(cc->intr_handle.fd, &e->msg, e->fd); 7009c7e63aSJakub Grajciar if (size != sizeof(memif_msg_t)) { 7109c7e63aSJakub Grajciar MIF_LOG(ERR, "sendmsg fail: %s.", strerror(errno)); 7209c7e63aSJakub Grajciar ret = -1; 7309c7e63aSJakub Grajciar } else { 7409c7e63aSJakub Grajciar MIF_LOG(DEBUG, "Sent msg type %u.", e->msg.type); 7509c7e63aSJakub Grajciar } 7609c7e63aSJakub Grajciar TAILQ_REMOVE(&cc->msg_queue, e, next); 7709c7e63aSJakub Grajciar rte_free(e); 7809c7e63aSJakub Grajciar 7909c7e63aSJakub Grajciar return ret; 8009c7e63aSJakub Grajciar } 8109c7e63aSJakub Grajciar 8209c7e63aSJakub Grajciar static struct memif_msg_queue_elt * 8309c7e63aSJakub Grajciar memif_msg_enq(struct memif_control_channel *cc) 8409c7e63aSJakub Grajciar { 8509c7e63aSJakub Grajciar struct memif_msg_queue_elt *e; 8609c7e63aSJakub Grajciar 8709c7e63aSJakub Grajciar e = rte_zmalloc("memif_msg", sizeof(struct memif_msg_queue_elt), 0); 8809c7e63aSJakub Grajciar if (e == NULL) { 8909c7e63aSJakub Grajciar MIF_LOG(ERR, "Failed to allocate control message."); 9009c7e63aSJakub Grajciar return NULL; 9109c7e63aSJakub Grajciar } 9209c7e63aSJakub Grajciar 9309c7e63aSJakub Grajciar e->fd = -1; 9409c7e63aSJakub Grajciar TAILQ_INSERT_TAIL(&cc->msg_queue, e, next); 9509c7e63aSJakub Grajciar 9609c7e63aSJakub Grajciar return e; 9709c7e63aSJakub Grajciar } 9809c7e63aSJakub Grajciar 9909c7e63aSJakub Grajciar void 10009c7e63aSJakub Grajciar memif_msg_enq_disconnect(struct memif_control_channel *cc, const char *reason, 10109c7e63aSJakub Grajciar int err_code) 10209c7e63aSJakub Grajciar { 10309c7e63aSJakub Grajciar struct memif_msg_queue_elt *e; 10409c7e63aSJakub Grajciar struct pmd_internals *pmd; 10509c7e63aSJakub Grajciar memif_msg_disconnect_t *d; 10609c7e63aSJakub Grajciar 10709c7e63aSJakub Grajciar if (cc == NULL) { 10809c7e63aSJakub Grajciar MIF_LOG(DEBUG, "Missing control channel."); 10909c7e63aSJakub Grajciar return; 11009c7e63aSJakub Grajciar } 11109c7e63aSJakub Grajciar 11209c7e63aSJakub Grajciar e = memif_msg_enq(cc); 11309c7e63aSJakub Grajciar if (e == NULL) { 11409c7e63aSJakub Grajciar MIF_LOG(WARNING, "Failed to enqueue disconnect message."); 11509c7e63aSJakub Grajciar return; 11609c7e63aSJakub Grajciar } 11709c7e63aSJakub Grajciar 11809c7e63aSJakub Grajciar d = &e->msg.disconnect; 11909c7e63aSJakub Grajciar 12009c7e63aSJakub Grajciar e->msg.type = MEMIF_MSG_TYPE_DISCONNECT; 12109c7e63aSJakub Grajciar d->code = err_code; 12209c7e63aSJakub Grajciar 12309c7e63aSJakub Grajciar if (reason != NULL) { 12409c7e63aSJakub Grajciar strlcpy((char *)d->string, reason, sizeof(d->string)); 12509c7e63aSJakub Grajciar if (cc->dev != NULL) { 12609c7e63aSJakub Grajciar pmd = cc->dev->data->dev_private; 12709c7e63aSJakub Grajciar strlcpy(pmd->local_disc_string, reason, 12809c7e63aSJakub Grajciar sizeof(pmd->local_disc_string)); 12909c7e63aSJakub Grajciar } 13009c7e63aSJakub Grajciar } 13109c7e63aSJakub Grajciar } 13209c7e63aSJakub Grajciar 13309c7e63aSJakub Grajciar static int 13409c7e63aSJakub Grajciar memif_msg_enq_hello(struct memif_control_channel *cc) 13509c7e63aSJakub Grajciar { 13609c7e63aSJakub Grajciar struct memif_msg_queue_elt *e = memif_msg_enq(cc); 13709c7e63aSJakub Grajciar memif_msg_hello_t *h; 13809c7e63aSJakub Grajciar 13909c7e63aSJakub Grajciar if (e == NULL) 14009c7e63aSJakub Grajciar return -1; 14109c7e63aSJakub Grajciar 14209c7e63aSJakub Grajciar h = &e->msg.hello; 14309c7e63aSJakub Grajciar 14409c7e63aSJakub Grajciar e->msg.type = MEMIF_MSG_TYPE_HELLO; 14509c7e63aSJakub Grajciar h->min_version = MEMIF_VERSION; 14609c7e63aSJakub Grajciar h->max_version = MEMIF_VERSION; 14709c7e63aSJakub Grajciar h->max_s2m_ring = ETH_MEMIF_MAX_NUM_Q_PAIRS; 14809c7e63aSJakub Grajciar h->max_m2s_ring = ETH_MEMIF_MAX_NUM_Q_PAIRS; 14909c7e63aSJakub Grajciar h->max_region = ETH_MEMIF_MAX_REGION_NUM - 1; 15009c7e63aSJakub Grajciar h->max_log2_ring_size = ETH_MEMIF_MAX_LOG2_RING_SIZE; 15109c7e63aSJakub Grajciar 15209c7e63aSJakub Grajciar strlcpy((char *)h->name, rte_version(), sizeof(h->name)); 15309c7e63aSJakub Grajciar 15409c7e63aSJakub Grajciar return 0; 15509c7e63aSJakub Grajciar } 15609c7e63aSJakub Grajciar 15709c7e63aSJakub Grajciar static int 15809c7e63aSJakub Grajciar memif_msg_receive_hello(struct rte_eth_dev *dev, memif_msg_t *msg) 15909c7e63aSJakub Grajciar { 16009c7e63aSJakub Grajciar struct pmd_internals *pmd = dev->data->dev_private; 16109c7e63aSJakub Grajciar memif_msg_hello_t *h = &msg->hello; 16209c7e63aSJakub Grajciar 16309c7e63aSJakub Grajciar if (h->min_version > MEMIF_VERSION || h->max_version < MEMIF_VERSION) { 16409c7e63aSJakub Grajciar memif_msg_enq_disconnect(pmd->cc, "Incompatible memif version", 0); 16509c7e63aSJakub Grajciar return -1; 16609c7e63aSJakub Grajciar } 16709c7e63aSJakub Grajciar 16809c7e63aSJakub Grajciar /* Set parameters for active connection */ 16909c7e63aSJakub Grajciar pmd->run.num_s2m_rings = RTE_MIN(h->max_s2m_ring + 1, 17009c7e63aSJakub Grajciar pmd->cfg.num_s2m_rings); 17109c7e63aSJakub Grajciar pmd->run.num_m2s_rings = RTE_MIN(h->max_m2s_ring + 1, 17209c7e63aSJakub Grajciar pmd->cfg.num_m2s_rings); 17309c7e63aSJakub Grajciar pmd->run.log2_ring_size = RTE_MIN(h->max_log2_ring_size, 17409c7e63aSJakub Grajciar pmd->cfg.log2_ring_size); 17509c7e63aSJakub Grajciar pmd->run.pkt_buffer_size = pmd->cfg.pkt_buffer_size; 17609c7e63aSJakub Grajciar 17709c7e63aSJakub Grajciar strlcpy(pmd->remote_name, (char *)h->name, sizeof(pmd->remote_name)); 17809c7e63aSJakub Grajciar 17909c7e63aSJakub Grajciar MIF_LOG(DEBUG, "%s: Connecting to %s.", 18009c7e63aSJakub Grajciar rte_vdev_device_name(pmd->vdev), pmd->remote_name); 18109c7e63aSJakub Grajciar 18209c7e63aSJakub Grajciar return 0; 18309c7e63aSJakub Grajciar } 18409c7e63aSJakub Grajciar 18509c7e63aSJakub Grajciar static int 18609c7e63aSJakub Grajciar memif_msg_receive_init(struct memif_control_channel *cc, memif_msg_t *msg) 18709c7e63aSJakub Grajciar { 18809c7e63aSJakub Grajciar memif_msg_init_t *i = &msg->init; 18909c7e63aSJakub Grajciar struct memif_socket_dev_list_elt *elt; 19009c7e63aSJakub Grajciar struct pmd_internals *pmd; 19109c7e63aSJakub Grajciar struct rte_eth_dev *dev; 19209c7e63aSJakub Grajciar 19309c7e63aSJakub Grajciar if (i->version != MEMIF_VERSION) { 19409c7e63aSJakub Grajciar memif_msg_enq_disconnect(cc, "Incompatible memif version", 0); 19509c7e63aSJakub Grajciar return -1; 19609c7e63aSJakub Grajciar } 19709c7e63aSJakub Grajciar 19809c7e63aSJakub Grajciar if (cc->socket == NULL) { 19909c7e63aSJakub Grajciar memif_msg_enq_disconnect(cc, "Device error", 0); 20009c7e63aSJakub Grajciar return -1; 20109c7e63aSJakub Grajciar } 20209c7e63aSJakub Grajciar 20309c7e63aSJakub Grajciar /* Find device with requested ID */ 20409c7e63aSJakub Grajciar TAILQ_FOREACH(elt, &cc->socket->dev_queue, next) { 20509c7e63aSJakub Grajciar dev = elt->dev; 20609c7e63aSJakub Grajciar pmd = dev->data->dev_private; 20709c7e63aSJakub Grajciar if (((pmd->flags & ETH_MEMIF_FLAG_DISABLED) == 0) && 20809c7e63aSJakub Grajciar pmd->id == i->id) { 20909c7e63aSJakub Grajciar /* assign control channel to device */ 21009c7e63aSJakub Grajciar cc->dev = dev; 21109c7e63aSJakub Grajciar pmd->cc = cc; 21209c7e63aSJakub Grajciar 21309c7e63aSJakub Grajciar if (i->mode != MEMIF_INTERFACE_MODE_ETHERNET) { 21409c7e63aSJakub Grajciar memif_msg_enq_disconnect(pmd->cc, 21509c7e63aSJakub Grajciar "Only ethernet mode supported", 21609c7e63aSJakub Grajciar 0); 21709c7e63aSJakub Grajciar return -1; 21809c7e63aSJakub Grajciar } 21909c7e63aSJakub Grajciar 22009c7e63aSJakub Grajciar if (pmd->flags & (ETH_MEMIF_FLAG_CONNECTING | 22109c7e63aSJakub Grajciar ETH_MEMIF_FLAG_CONNECTED)) { 22209c7e63aSJakub Grajciar memif_msg_enq_disconnect(pmd->cc, 22309c7e63aSJakub Grajciar "Already connected", 0); 22409c7e63aSJakub Grajciar return -1; 22509c7e63aSJakub Grajciar } 22609c7e63aSJakub Grajciar strlcpy(pmd->remote_name, (char *)i->name, 22709c7e63aSJakub Grajciar sizeof(pmd->remote_name)); 22809c7e63aSJakub Grajciar 22909c7e63aSJakub Grajciar if (*pmd->secret != '\0') { 23009c7e63aSJakub Grajciar if (*i->secret == '\0') { 23109c7e63aSJakub Grajciar memif_msg_enq_disconnect(pmd->cc, 23209c7e63aSJakub Grajciar "Secret required", 0); 23309c7e63aSJakub Grajciar return -1; 23409c7e63aSJakub Grajciar } 23509c7e63aSJakub Grajciar if (strncmp(pmd->secret, (char *)i->secret, 23609c7e63aSJakub Grajciar ETH_MEMIF_SECRET_SIZE) != 0) { 23709c7e63aSJakub Grajciar memif_msg_enq_disconnect(pmd->cc, 23809c7e63aSJakub Grajciar "Incorrect secret", 0); 23909c7e63aSJakub Grajciar return -1; 24009c7e63aSJakub Grajciar } 24109c7e63aSJakub Grajciar } 24209c7e63aSJakub Grajciar 24309c7e63aSJakub Grajciar pmd->flags |= ETH_MEMIF_FLAG_CONNECTING; 24409c7e63aSJakub Grajciar return 0; 24509c7e63aSJakub Grajciar } 24609c7e63aSJakub Grajciar } 24709c7e63aSJakub Grajciar 24809c7e63aSJakub Grajciar /* ID not found on this socket */ 24909c7e63aSJakub Grajciar MIF_LOG(DEBUG, "ID %u not found.", i->id); 25009c7e63aSJakub Grajciar memif_msg_enq_disconnect(cc, "ID not found", 0); 25109c7e63aSJakub Grajciar return -1; 25209c7e63aSJakub Grajciar } 25309c7e63aSJakub Grajciar 25409c7e63aSJakub Grajciar static int 25509c7e63aSJakub Grajciar memif_msg_receive_add_region(struct rte_eth_dev *dev, memif_msg_t *msg, 25609c7e63aSJakub Grajciar int fd) 25709c7e63aSJakub Grajciar { 25809c7e63aSJakub Grajciar struct pmd_internals *pmd = dev->data->dev_private; 259c41a0495SJakub Grajciar struct pmd_process_private *proc_private = dev->process_private; 26009c7e63aSJakub Grajciar memif_msg_add_region_t *ar = &msg->add_region; 26109c7e63aSJakub Grajciar struct memif_region *r; 26209c7e63aSJakub Grajciar 26309c7e63aSJakub Grajciar if (fd < 0) { 26409c7e63aSJakub Grajciar memif_msg_enq_disconnect(pmd->cc, "Missing region fd", 0); 26509c7e63aSJakub Grajciar return -1; 26609c7e63aSJakub Grajciar } 26709c7e63aSJakub Grajciar 268c41a0495SJakub Grajciar if (ar->index >= ETH_MEMIF_MAX_REGION_NUM || 269c41a0495SJakub Grajciar ar->index != proc_private->regions_num || 270c41a0495SJakub Grajciar proc_private->regions[ar->index] != NULL) { 27109c7e63aSJakub Grajciar memif_msg_enq_disconnect(pmd->cc, "Invalid region index", 0); 27209c7e63aSJakub Grajciar return -1; 27309c7e63aSJakub Grajciar } 27409c7e63aSJakub Grajciar 27509c7e63aSJakub Grajciar r = rte_zmalloc("region", sizeof(struct memif_region), 0); 27609c7e63aSJakub Grajciar if (r == NULL) { 277c41a0495SJakub Grajciar memif_msg_enq_disconnect(pmd->cc, "Failed to alloc memif region.", 0); 27809c7e63aSJakub Grajciar return -ENOMEM; 27909c7e63aSJakub Grajciar } 28009c7e63aSJakub Grajciar 28109c7e63aSJakub Grajciar r->fd = fd; 28209c7e63aSJakub Grajciar r->region_size = ar->size; 28309c7e63aSJakub Grajciar r->addr = NULL; 28409c7e63aSJakub Grajciar 285c41a0495SJakub Grajciar proc_private->regions[ar->index] = r; 286c41a0495SJakub Grajciar proc_private->regions_num++; 28709c7e63aSJakub Grajciar 28809c7e63aSJakub Grajciar return 0; 28909c7e63aSJakub Grajciar } 29009c7e63aSJakub Grajciar 29109c7e63aSJakub Grajciar static int 29209c7e63aSJakub Grajciar memif_msg_receive_add_ring(struct rte_eth_dev *dev, memif_msg_t *msg, int fd) 29309c7e63aSJakub Grajciar { 29409c7e63aSJakub Grajciar struct pmd_internals *pmd = dev->data->dev_private; 29509c7e63aSJakub Grajciar memif_msg_add_ring_t *ar = &msg->add_ring; 29609c7e63aSJakub Grajciar struct memif_queue *mq; 29709c7e63aSJakub Grajciar 29809c7e63aSJakub Grajciar if (fd < 0) { 29909c7e63aSJakub Grajciar memif_msg_enq_disconnect(pmd->cc, "Missing interrupt fd", 0); 30009c7e63aSJakub Grajciar return -1; 30109c7e63aSJakub Grajciar } 30209c7e63aSJakub Grajciar 30309c7e63aSJakub Grajciar /* check if we have enough queues */ 30409c7e63aSJakub Grajciar if (ar->flags & MEMIF_MSG_ADD_RING_FLAG_S2M) { 30509c7e63aSJakub Grajciar if (ar->index >= pmd->cfg.num_s2m_rings) { 30609c7e63aSJakub Grajciar memif_msg_enq_disconnect(pmd->cc, "Invalid ring index", 0); 30709c7e63aSJakub Grajciar return -1; 30809c7e63aSJakub Grajciar } 30909c7e63aSJakub Grajciar pmd->run.num_s2m_rings++; 31009c7e63aSJakub Grajciar } else { 31109c7e63aSJakub Grajciar if (ar->index >= pmd->cfg.num_m2s_rings) { 31209c7e63aSJakub Grajciar memif_msg_enq_disconnect(pmd->cc, "Invalid ring index", 0); 31309c7e63aSJakub Grajciar return -1; 31409c7e63aSJakub Grajciar } 31509c7e63aSJakub Grajciar pmd->run.num_m2s_rings++; 31609c7e63aSJakub Grajciar } 31709c7e63aSJakub Grajciar 31809c7e63aSJakub Grajciar mq = (ar->flags & MEMIF_MSG_ADD_RING_FLAG_S2M) ? 31909c7e63aSJakub Grajciar dev->data->rx_queues[ar->index] : dev->data->tx_queues[ar->index]; 32009c7e63aSJakub Grajciar 32109c7e63aSJakub Grajciar mq->intr_handle.fd = fd; 32209c7e63aSJakub Grajciar mq->log2_ring_size = ar->log2_ring_size; 32309c7e63aSJakub Grajciar mq->region = ar->region; 32409c7e63aSJakub Grajciar mq->ring_offset = ar->offset; 32509c7e63aSJakub Grajciar 32609c7e63aSJakub Grajciar return 0; 32709c7e63aSJakub Grajciar } 32809c7e63aSJakub Grajciar 32909c7e63aSJakub Grajciar static int 33009c7e63aSJakub Grajciar memif_msg_receive_connect(struct rte_eth_dev *dev, memif_msg_t *msg) 33109c7e63aSJakub Grajciar { 33209c7e63aSJakub Grajciar struct pmd_internals *pmd = dev->data->dev_private; 33309c7e63aSJakub Grajciar memif_msg_connect_t *c = &msg->connect; 33409c7e63aSJakub Grajciar int ret; 33509c7e63aSJakub Grajciar 33609c7e63aSJakub Grajciar ret = memif_connect(dev); 33709c7e63aSJakub Grajciar if (ret < 0) 33809c7e63aSJakub Grajciar return ret; 33909c7e63aSJakub Grajciar 34009c7e63aSJakub Grajciar strlcpy(pmd->remote_if_name, (char *)c->if_name, 34109c7e63aSJakub Grajciar sizeof(pmd->remote_if_name)); 34209c7e63aSJakub Grajciar MIF_LOG(INFO, "%s: Remote interface %s connected.", 34309c7e63aSJakub Grajciar rte_vdev_device_name(pmd->vdev), pmd->remote_if_name); 34409c7e63aSJakub Grajciar 34509c7e63aSJakub Grajciar return 0; 34609c7e63aSJakub Grajciar } 34709c7e63aSJakub Grajciar 34809c7e63aSJakub Grajciar static int 34909c7e63aSJakub Grajciar memif_msg_receive_connected(struct rte_eth_dev *dev, memif_msg_t *msg) 35009c7e63aSJakub Grajciar { 35109c7e63aSJakub Grajciar struct pmd_internals *pmd = dev->data->dev_private; 35209c7e63aSJakub Grajciar memif_msg_connected_t *c = &msg->connected; 35309c7e63aSJakub Grajciar int ret; 35409c7e63aSJakub Grajciar 35509c7e63aSJakub Grajciar ret = memif_connect(dev); 35609c7e63aSJakub Grajciar if (ret < 0) 35709c7e63aSJakub Grajciar return ret; 35809c7e63aSJakub Grajciar 35909c7e63aSJakub Grajciar strlcpy(pmd->remote_if_name, (char *)c->if_name, 36009c7e63aSJakub Grajciar sizeof(pmd->remote_if_name)); 36109c7e63aSJakub Grajciar MIF_LOG(INFO, "%s: Remote interface %s connected.", 36209c7e63aSJakub Grajciar rte_vdev_device_name(pmd->vdev), pmd->remote_if_name); 36309c7e63aSJakub Grajciar 36409c7e63aSJakub Grajciar return 0; 36509c7e63aSJakub Grajciar } 36609c7e63aSJakub Grajciar 36709c7e63aSJakub Grajciar static int 36809c7e63aSJakub Grajciar memif_msg_receive_disconnect(struct rte_eth_dev *dev, memif_msg_t *msg) 36909c7e63aSJakub Grajciar { 37009c7e63aSJakub Grajciar struct pmd_internals *pmd = dev->data->dev_private; 37109c7e63aSJakub Grajciar memif_msg_disconnect_t *d = &msg->disconnect; 37209c7e63aSJakub Grajciar 37309c7e63aSJakub Grajciar memset(pmd->remote_disc_string, 0, ETH_MEMIF_DISC_STRING_SIZE); 37409c7e63aSJakub Grajciar strlcpy(pmd->remote_disc_string, (char *)d->string, 37509c7e63aSJakub Grajciar sizeof(pmd->remote_disc_string)); 37609c7e63aSJakub Grajciar 37709c7e63aSJakub Grajciar MIF_LOG(INFO, "%s: Disconnect received: %s", 37809c7e63aSJakub Grajciar rte_vdev_device_name(pmd->vdev), pmd->remote_disc_string); 37909c7e63aSJakub Grajciar 38009c7e63aSJakub Grajciar memset(pmd->local_disc_string, 0, ETH_MEMIF_DISC_STRING_SIZE); 381c41a0495SJakub Grajciar memif_disconnect(dev); 38209c7e63aSJakub Grajciar return 0; 38309c7e63aSJakub Grajciar } 38409c7e63aSJakub Grajciar 38509c7e63aSJakub Grajciar static int 38609c7e63aSJakub Grajciar memif_msg_enq_ack(struct rte_eth_dev *dev) 38709c7e63aSJakub Grajciar { 38809c7e63aSJakub Grajciar struct pmd_internals *pmd = dev->data->dev_private; 38909c7e63aSJakub Grajciar struct memif_msg_queue_elt *e = memif_msg_enq(pmd->cc); 39009c7e63aSJakub Grajciar if (e == NULL) 39109c7e63aSJakub Grajciar return -1; 39209c7e63aSJakub Grajciar 39309c7e63aSJakub Grajciar e->msg.type = MEMIF_MSG_TYPE_ACK; 39409c7e63aSJakub Grajciar 39509c7e63aSJakub Grajciar return 0; 39609c7e63aSJakub Grajciar } 39709c7e63aSJakub Grajciar 39809c7e63aSJakub Grajciar static int 39909c7e63aSJakub Grajciar memif_msg_enq_init(struct rte_eth_dev *dev) 40009c7e63aSJakub Grajciar { 40109c7e63aSJakub Grajciar struct pmd_internals *pmd = dev->data->dev_private; 40209c7e63aSJakub Grajciar struct memif_msg_queue_elt *e = memif_msg_enq(pmd->cc); 40309c7e63aSJakub Grajciar memif_msg_init_t *i = &e->msg.init; 40409c7e63aSJakub Grajciar 40509c7e63aSJakub Grajciar if (e == NULL) 40609c7e63aSJakub Grajciar return -1; 40709c7e63aSJakub Grajciar 40809c7e63aSJakub Grajciar i = &e->msg.init; 40909c7e63aSJakub Grajciar e->msg.type = MEMIF_MSG_TYPE_INIT; 41009c7e63aSJakub Grajciar i->version = MEMIF_VERSION; 41109c7e63aSJakub Grajciar i->id = pmd->id; 41209c7e63aSJakub Grajciar i->mode = MEMIF_INTERFACE_MODE_ETHERNET; 41309c7e63aSJakub Grajciar 41409c7e63aSJakub Grajciar strlcpy((char *)i->name, rte_version(), sizeof(i->name)); 41509c7e63aSJakub Grajciar 41609c7e63aSJakub Grajciar if (*pmd->secret != '\0') 41709c7e63aSJakub Grajciar strlcpy((char *)i->secret, pmd->secret, sizeof(i->secret)); 41809c7e63aSJakub Grajciar 41909c7e63aSJakub Grajciar return 0; 42009c7e63aSJakub Grajciar } 42109c7e63aSJakub Grajciar 42209c7e63aSJakub Grajciar static int 42309c7e63aSJakub Grajciar memif_msg_enq_add_region(struct rte_eth_dev *dev, uint8_t idx) 42409c7e63aSJakub Grajciar { 42509c7e63aSJakub Grajciar struct pmd_internals *pmd = dev->data->dev_private; 426c41a0495SJakub Grajciar struct pmd_process_private *proc_private = dev->process_private; 42709c7e63aSJakub Grajciar struct memif_msg_queue_elt *e = memif_msg_enq(pmd->cc); 42809c7e63aSJakub Grajciar memif_msg_add_region_t *ar; 429c41a0495SJakub Grajciar struct memif_region *mr = proc_private->regions[idx]; 43009c7e63aSJakub Grajciar 43109c7e63aSJakub Grajciar if (e == NULL) 43209c7e63aSJakub Grajciar return -1; 43309c7e63aSJakub Grajciar 43409c7e63aSJakub Grajciar ar = &e->msg.add_region; 43509c7e63aSJakub Grajciar e->msg.type = MEMIF_MSG_TYPE_ADD_REGION; 43609c7e63aSJakub Grajciar e->fd = mr->fd; 43709c7e63aSJakub Grajciar ar->index = idx; 43809c7e63aSJakub Grajciar ar->size = mr->region_size; 43909c7e63aSJakub Grajciar 44009c7e63aSJakub Grajciar return 0; 44109c7e63aSJakub Grajciar } 44209c7e63aSJakub Grajciar 44309c7e63aSJakub Grajciar static int 44409c7e63aSJakub Grajciar memif_msg_enq_add_ring(struct rte_eth_dev *dev, uint8_t idx, 44509c7e63aSJakub Grajciar memif_ring_type_t type) 44609c7e63aSJakub Grajciar { 44709c7e63aSJakub Grajciar struct pmd_internals *pmd = dev->data->dev_private; 44809c7e63aSJakub Grajciar struct memif_msg_queue_elt *e = memif_msg_enq(pmd->cc); 44909c7e63aSJakub Grajciar struct memif_queue *mq; 45009c7e63aSJakub Grajciar memif_msg_add_ring_t *ar; 45109c7e63aSJakub Grajciar 45209c7e63aSJakub Grajciar if (e == NULL) 45309c7e63aSJakub Grajciar return -1; 45409c7e63aSJakub Grajciar 45509c7e63aSJakub Grajciar ar = &e->msg.add_ring; 45609c7e63aSJakub Grajciar mq = (type == MEMIF_RING_S2M) ? dev->data->tx_queues[idx] : 45709c7e63aSJakub Grajciar dev->data->rx_queues[idx]; 45809c7e63aSJakub Grajciar 45909c7e63aSJakub Grajciar e->msg.type = MEMIF_MSG_TYPE_ADD_RING; 46009c7e63aSJakub Grajciar e->fd = mq->intr_handle.fd; 46109c7e63aSJakub Grajciar ar->index = idx; 46209c7e63aSJakub Grajciar ar->offset = mq->ring_offset; 46309c7e63aSJakub Grajciar ar->region = mq->region; 46409c7e63aSJakub Grajciar ar->log2_ring_size = mq->log2_ring_size; 46509c7e63aSJakub Grajciar ar->flags = (type == MEMIF_RING_S2M) ? MEMIF_MSG_ADD_RING_FLAG_S2M : 0; 46609c7e63aSJakub Grajciar ar->private_hdr_size = 0; 46709c7e63aSJakub Grajciar 46809c7e63aSJakub Grajciar return 0; 46909c7e63aSJakub Grajciar } 47009c7e63aSJakub Grajciar 47109c7e63aSJakub Grajciar static int 47209c7e63aSJakub Grajciar memif_msg_enq_connect(struct rte_eth_dev *dev) 47309c7e63aSJakub Grajciar { 47409c7e63aSJakub Grajciar struct pmd_internals *pmd = dev->data->dev_private; 47509c7e63aSJakub Grajciar struct memif_msg_queue_elt *e = memif_msg_enq(pmd->cc); 47609c7e63aSJakub Grajciar const char *name = rte_vdev_device_name(pmd->vdev); 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; 48409c7e63aSJakub Grajciar strlcpy((char *)c->if_name, name, sizeof(c->if_name)); 48509c7e63aSJakub Grajciar 48609c7e63aSJakub Grajciar return 0; 48709c7e63aSJakub Grajciar } 48809c7e63aSJakub Grajciar 48909c7e63aSJakub Grajciar static int 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 const char *name = rte_vdev_device_name(pmd->vdev); 49509c7e63aSJakub Grajciar memif_msg_connected_t *c; 49609c7e63aSJakub Grajciar 49709c7e63aSJakub Grajciar if (e == NULL) 49809c7e63aSJakub Grajciar return -1; 49909c7e63aSJakub Grajciar 50009c7e63aSJakub Grajciar c = &e->msg.connected; 50109c7e63aSJakub Grajciar e->msg.type = MEMIF_MSG_TYPE_CONNECTED; 50209c7e63aSJakub Grajciar strlcpy((char *)c->if_name, name, sizeof(c->if_name)); 50309c7e63aSJakub Grajciar 50409c7e63aSJakub Grajciar return 0; 50509c7e63aSJakub Grajciar } 50609c7e63aSJakub Grajciar 50709c7e63aSJakub Grajciar static void 50809c7e63aSJakub Grajciar memif_intr_unregister_handler(struct rte_intr_handle *intr_handle, void *arg) 50909c7e63aSJakub Grajciar { 51009c7e63aSJakub Grajciar struct memif_msg_queue_elt *elt; 51109c7e63aSJakub Grajciar struct memif_control_channel *cc = arg; 51209c7e63aSJakub Grajciar 51309c7e63aSJakub Grajciar /* close control channel fd */ 51409c7e63aSJakub Grajciar close(intr_handle->fd); 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 } 52009c7e63aSJakub Grajciar /* free control channel */ 52109c7e63aSJakub Grajciar rte_free(cc); 52209c7e63aSJakub Grajciar } 52309c7e63aSJakub Grajciar 52409c7e63aSJakub Grajciar void 52509c7e63aSJakub Grajciar memif_disconnect(struct rte_eth_dev *dev) 52609c7e63aSJakub Grajciar { 52709c7e63aSJakub Grajciar struct pmd_internals *pmd = dev->data->dev_private; 528c41a0495SJakub Grajciar struct pmd_process_private *proc_private = dev->process_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 535c41a0495SJakub Grajciar dev->data->dev_link.link_status = ETH_LINK_DOWN; 536c41a0495SJakub Grajciar pmd->flags &= ~ETH_MEMIF_FLAG_CONNECTING; 537c41a0495SJakub Grajciar pmd->flags &= ~ETH_MEMIF_FLAG_CONNECTED; 538c41a0495SJakub Grajciar 53909c7e63aSJakub Grajciar if (pmd->cc != NULL) { 54009c7e63aSJakub Grajciar /* Clear control message queue (except disconnect message if any). */ 54109c7e63aSJakub Grajciar for (elt = TAILQ_FIRST(&pmd->cc->msg_queue); elt != NULL; elt = next) { 54209c7e63aSJakub Grajciar next = TAILQ_NEXT(elt, next); 54309c7e63aSJakub Grajciar if (elt->msg.type != MEMIF_MSG_TYPE_DISCONNECT) { 54409c7e63aSJakub Grajciar TAILQ_REMOVE(&pmd->cc->msg_queue, elt, next); 54509c7e63aSJakub Grajciar rte_free(elt); 54609c7e63aSJakub Grajciar } 54709c7e63aSJakub Grajciar } 54809c7e63aSJakub Grajciar /* send disconnect message (if there is any in queue) */ 54909c7e63aSJakub Grajciar memif_msg_send_from_queue(pmd->cc); 55009c7e63aSJakub Grajciar 55109c7e63aSJakub Grajciar /* at this point, there should be no more messages in queue */ 55209c7e63aSJakub Grajciar if (TAILQ_FIRST(&pmd->cc->msg_queue) != NULL) { 55309c7e63aSJakub Grajciar MIF_LOG(WARNING, 554c41a0495SJakub Grajciar "Unexpected message(s) in message queue."); 55509c7e63aSJakub Grajciar } 55609c7e63aSJakub Grajciar 55709c7e63aSJakub Grajciar ih = &pmd->cc->intr_handle; 55809c7e63aSJakub Grajciar if (ih->fd > 0) { 55909c7e63aSJakub Grajciar ret = rte_intr_callback_unregister(ih, 56009c7e63aSJakub Grajciar memif_intr_handler, 56109c7e63aSJakub Grajciar pmd->cc); 56209c7e63aSJakub Grajciar /* 56309c7e63aSJakub Grajciar * If callback is active (disconnecting based on 56409c7e63aSJakub Grajciar * received control message). 56509c7e63aSJakub Grajciar */ 56609c7e63aSJakub Grajciar if (ret == -EAGAIN) { 56709c7e63aSJakub Grajciar ret = rte_intr_callback_unregister_pending(ih, 56809c7e63aSJakub Grajciar memif_intr_handler, 56909c7e63aSJakub Grajciar pmd->cc, 57009c7e63aSJakub Grajciar memif_intr_unregister_handler); 57109c7e63aSJakub Grajciar } else if (ret > 0) { 57209c7e63aSJakub Grajciar close(ih->fd); 57309c7e63aSJakub Grajciar rte_free(pmd->cc); 57409c7e63aSJakub Grajciar } 57509c7e63aSJakub Grajciar pmd->cc = NULL; 57609c7e63aSJakub Grajciar if (ret <= 0) 577c41a0495SJakub Grajciar MIF_LOG(WARNING, 578c41a0495SJakub Grajciar "Failed to unregister control channel callback."); 57909c7e63aSJakub Grajciar } 58009c7e63aSJakub Grajciar } 58109c7e63aSJakub Grajciar 58209c7e63aSJakub Grajciar /* unconfig interrupts */ 58309c7e63aSJakub Grajciar for (i = 0; i < pmd->cfg.num_s2m_rings; i++) { 58409c7e63aSJakub Grajciar if (pmd->role == MEMIF_ROLE_SLAVE) { 58509c7e63aSJakub Grajciar if (dev->data->tx_queues != NULL) 58609c7e63aSJakub Grajciar mq = dev->data->tx_queues[i]; 58709c7e63aSJakub Grajciar else 58809c7e63aSJakub Grajciar continue; 58909c7e63aSJakub Grajciar } else { 59009c7e63aSJakub Grajciar if (dev->data->rx_queues != NULL) 59109c7e63aSJakub Grajciar mq = dev->data->rx_queues[i]; 59209c7e63aSJakub Grajciar else 59309c7e63aSJakub Grajciar continue; 59409c7e63aSJakub Grajciar } 59509c7e63aSJakub Grajciar if (mq->intr_handle.fd > 0) { 59609c7e63aSJakub Grajciar close(mq->intr_handle.fd); 59709c7e63aSJakub Grajciar mq->intr_handle.fd = -1; 59809c7e63aSJakub Grajciar } 59909c7e63aSJakub Grajciar } 60009c7e63aSJakub Grajciar for (i = 0; i < pmd->cfg.num_m2s_rings; i++) { 60109c7e63aSJakub Grajciar if (pmd->role == MEMIF_ROLE_MASTER) { 60209c7e63aSJakub Grajciar if (dev->data->tx_queues != NULL) 60309c7e63aSJakub Grajciar mq = dev->data->tx_queues[i]; 60409c7e63aSJakub Grajciar else 60509c7e63aSJakub Grajciar continue; 60609c7e63aSJakub Grajciar } else { 60709c7e63aSJakub Grajciar if (dev->data->rx_queues != NULL) 60809c7e63aSJakub Grajciar mq = dev->data->rx_queues[i]; 60909c7e63aSJakub Grajciar else 61009c7e63aSJakub Grajciar continue; 61109c7e63aSJakub Grajciar } 61209c7e63aSJakub Grajciar if (mq->intr_handle.fd > 0) { 61309c7e63aSJakub Grajciar close(mq->intr_handle.fd); 61409c7e63aSJakub Grajciar mq->intr_handle.fd = -1; 61509c7e63aSJakub Grajciar } 61609c7e63aSJakub Grajciar } 61709c7e63aSJakub Grajciar 618c41a0495SJakub Grajciar memif_free_regions(proc_private); 61909c7e63aSJakub Grajciar 62009c7e63aSJakub Grajciar /* reset connection configuration */ 62109c7e63aSJakub Grajciar memset(&pmd->run, 0, sizeof(pmd->run)); 62209c7e63aSJakub Grajciar 623c41a0495SJakub Grajciar MIF_LOG(DEBUG, "Disconnected."); 62409c7e63aSJakub Grajciar } 62509c7e63aSJakub Grajciar 62609c7e63aSJakub Grajciar static int 62709c7e63aSJakub Grajciar memif_msg_receive(struct memif_control_channel *cc) 62809c7e63aSJakub Grajciar { 62909c7e63aSJakub Grajciar char ctl[CMSG_SPACE(sizeof(int)) + 63009c7e63aSJakub Grajciar CMSG_SPACE(sizeof(struct ucred))] = { 0 }; 63109c7e63aSJakub Grajciar struct msghdr mh = { 0 }; 63209c7e63aSJakub Grajciar struct iovec iov[1]; 63309c7e63aSJakub Grajciar memif_msg_t msg = { 0 }; 63409c7e63aSJakub Grajciar ssize_t size; 63509c7e63aSJakub Grajciar int ret = 0; 63609c7e63aSJakub Grajciar struct ucred *cr __rte_unused; 63709c7e63aSJakub Grajciar cr = 0; 63809c7e63aSJakub Grajciar struct cmsghdr *cmsg; 63909c7e63aSJakub Grajciar int afd = -1; 64009c7e63aSJakub Grajciar int i; 64109c7e63aSJakub Grajciar struct pmd_internals *pmd; 642c41a0495SJakub Grajciar struct pmd_process_private *proc_private; 64309c7e63aSJakub Grajciar 64409c7e63aSJakub Grajciar iov[0].iov_base = (void *)&msg; 64509c7e63aSJakub Grajciar iov[0].iov_len = sizeof(memif_msg_t); 64609c7e63aSJakub Grajciar mh.msg_iov = iov; 64709c7e63aSJakub Grajciar mh.msg_iovlen = 1; 64809c7e63aSJakub Grajciar mh.msg_control = ctl; 64909c7e63aSJakub Grajciar mh.msg_controllen = sizeof(ctl); 65009c7e63aSJakub Grajciar 65109c7e63aSJakub Grajciar size = recvmsg(cc->intr_handle.fd, &mh, 0); 65209c7e63aSJakub Grajciar if (size != sizeof(memif_msg_t)) { 65309c7e63aSJakub Grajciar MIF_LOG(DEBUG, "Invalid message size."); 65409c7e63aSJakub Grajciar memif_msg_enq_disconnect(cc, "Invalid message size", 0); 65509c7e63aSJakub Grajciar return -1; 65609c7e63aSJakub Grajciar } 65709c7e63aSJakub Grajciar MIF_LOG(DEBUG, "Received msg type: %u.", msg.type); 65809c7e63aSJakub Grajciar 65909c7e63aSJakub Grajciar cmsg = CMSG_FIRSTHDR(&mh); 66009c7e63aSJakub Grajciar while (cmsg) { 66109c7e63aSJakub Grajciar if (cmsg->cmsg_level == SOL_SOCKET) { 66209c7e63aSJakub Grajciar if (cmsg->cmsg_type == SCM_CREDENTIALS) 66309c7e63aSJakub Grajciar cr = (struct ucred *)CMSG_DATA(cmsg); 66409c7e63aSJakub Grajciar else if (cmsg->cmsg_type == SCM_RIGHTS) 66509c7e63aSJakub Grajciar memcpy(&afd, CMSG_DATA(cmsg), sizeof(int)); 66609c7e63aSJakub Grajciar } 66709c7e63aSJakub Grajciar cmsg = CMSG_NXTHDR(&mh, cmsg); 66809c7e63aSJakub Grajciar } 66909c7e63aSJakub Grajciar 67009c7e63aSJakub Grajciar if (cc->dev == NULL && msg.type != MEMIF_MSG_TYPE_INIT) { 67109c7e63aSJakub Grajciar MIF_LOG(DEBUG, "Unexpected message."); 67209c7e63aSJakub Grajciar memif_msg_enq_disconnect(cc, "Unexpected message", 0); 67309c7e63aSJakub Grajciar return -1; 67409c7e63aSJakub Grajciar } 67509c7e63aSJakub Grajciar 67609c7e63aSJakub Grajciar /* get device from hash data */ 67709c7e63aSJakub Grajciar switch (msg.type) { 67809c7e63aSJakub Grajciar case MEMIF_MSG_TYPE_ACK: 67909c7e63aSJakub Grajciar break; 68009c7e63aSJakub Grajciar case MEMIF_MSG_TYPE_HELLO: 68109c7e63aSJakub Grajciar ret = memif_msg_receive_hello(cc->dev, &msg); 68209c7e63aSJakub Grajciar if (ret < 0) 68309c7e63aSJakub Grajciar goto exit; 68409c7e63aSJakub Grajciar ret = memif_init_regions_and_queues(cc->dev); 68509c7e63aSJakub Grajciar if (ret < 0) 68609c7e63aSJakub Grajciar goto exit; 68709c7e63aSJakub Grajciar ret = memif_msg_enq_init(cc->dev); 68809c7e63aSJakub Grajciar if (ret < 0) 68909c7e63aSJakub Grajciar goto exit; 69009c7e63aSJakub Grajciar pmd = cc->dev->data->dev_private; 691c41a0495SJakub Grajciar proc_private = cc->dev->process_private; 692c41a0495SJakub Grajciar for (i = 0; i < proc_private->regions_num; i++) { 69309c7e63aSJakub Grajciar ret = memif_msg_enq_add_region(cc->dev, i); 69409c7e63aSJakub Grajciar if (ret < 0) 69509c7e63aSJakub Grajciar goto exit; 69609c7e63aSJakub Grajciar } 69709c7e63aSJakub Grajciar for (i = 0; i < pmd->run.num_s2m_rings; i++) { 69809c7e63aSJakub Grajciar ret = memif_msg_enq_add_ring(cc->dev, i, 69909c7e63aSJakub Grajciar MEMIF_RING_S2M); 70009c7e63aSJakub Grajciar if (ret < 0) 70109c7e63aSJakub Grajciar goto exit; 70209c7e63aSJakub Grajciar } 70309c7e63aSJakub Grajciar for (i = 0; i < pmd->run.num_m2s_rings; i++) { 70409c7e63aSJakub Grajciar ret = memif_msg_enq_add_ring(cc->dev, i, 70509c7e63aSJakub Grajciar MEMIF_RING_M2S); 70609c7e63aSJakub Grajciar if (ret < 0) 70709c7e63aSJakub Grajciar goto exit; 70809c7e63aSJakub Grajciar } 70909c7e63aSJakub Grajciar ret = memif_msg_enq_connect(cc->dev); 71009c7e63aSJakub Grajciar if (ret < 0) 71109c7e63aSJakub Grajciar goto exit; 71209c7e63aSJakub Grajciar break; 71309c7e63aSJakub Grajciar case MEMIF_MSG_TYPE_INIT: 71409c7e63aSJakub Grajciar /* 71509c7e63aSJakub Grajciar * This cc does not have an interface asociated with it. 71609c7e63aSJakub Grajciar * If suitable interface is found it will be assigned here. 71709c7e63aSJakub Grajciar */ 71809c7e63aSJakub Grajciar ret = memif_msg_receive_init(cc, &msg); 71909c7e63aSJakub Grajciar if (ret < 0) 72009c7e63aSJakub Grajciar goto exit; 72109c7e63aSJakub Grajciar ret = memif_msg_enq_ack(cc->dev); 72209c7e63aSJakub Grajciar if (ret < 0) 72309c7e63aSJakub Grajciar goto exit; 72409c7e63aSJakub Grajciar break; 72509c7e63aSJakub Grajciar case MEMIF_MSG_TYPE_ADD_REGION: 72609c7e63aSJakub Grajciar ret = memif_msg_receive_add_region(cc->dev, &msg, afd); 72709c7e63aSJakub Grajciar if (ret < 0) 72809c7e63aSJakub Grajciar goto exit; 72909c7e63aSJakub Grajciar ret = memif_msg_enq_ack(cc->dev); 73009c7e63aSJakub Grajciar if (ret < 0) 73109c7e63aSJakub Grajciar goto exit; 73209c7e63aSJakub Grajciar break; 73309c7e63aSJakub Grajciar case MEMIF_MSG_TYPE_ADD_RING: 73409c7e63aSJakub Grajciar ret = memif_msg_receive_add_ring(cc->dev, &msg, afd); 73509c7e63aSJakub Grajciar if (ret < 0) 73609c7e63aSJakub Grajciar goto exit; 73709c7e63aSJakub Grajciar ret = memif_msg_enq_ack(cc->dev); 73809c7e63aSJakub Grajciar if (ret < 0) 73909c7e63aSJakub Grajciar goto exit; 74009c7e63aSJakub Grajciar break; 74109c7e63aSJakub Grajciar case MEMIF_MSG_TYPE_CONNECT: 74209c7e63aSJakub Grajciar ret = memif_msg_receive_connect(cc->dev, &msg); 74309c7e63aSJakub Grajciar if (ret < 0) 74409c7e63aSJakub Grajciar goto exit; 74509c7e63aSJakub Grajciar ret = memif_msg_enq_connected(cc->dev); 74609c7e63aSJakub Grajciar if (ret < 0) 74709c7e63aSJakub Grajciar goto exit; 74809c7e63aSJakub Grajciar break; 74909c7e63aSJakub Grajciar case MEMIF_MSG_TYPE_CONNECTED: 75009c7e63aSJakub Grajciar ret = memif_msg_receive_connected(cc->dev, &msg); 75109c7e63aSJakub Grajciar break; 75209c7e63aSJakub Grajciar case MEMIF_MSG_TYPE_DISCONNECT: 75309c7e63aSJakub Grajciar ret = memif_msg_receive_disconnect(cc->dev, &msg); 75409c7e63aSJakub Grajciar if (ret < 0) 75509c7e63aSJakub Grajciar goto exit; 75609c7e63aSJakub Grajciar break; 75709c7e63aSJakub Grajciar default: 75809c7e63aSJakub Grajciar memif_msg_enq_disconnect(cc, "Unknown message type", 0); 75909c7e63aSJakub Grajciar ret = -1; 76009c7e63aSJakub Grajciar goto exit; 76109c7e63aSJakub Grajciar } 76209c7e63aSJakub Grajciar 76309c7e63aSJakub Grajciar exit: 76409c7e63aSJakub Grajciar return ret; 76509c7e63aSJakub Grajciar } 76609c7e63aSJakub Grajciar 76709c7e63aSJakub Grajciar static void 76809c7e63aSJakub Grajciar memif_intr_handler(void *arg) 76909c7e63aSJakub Grajciar { 77009c7e63aSJakub Grajciar struct memif_control_channel *cc = arg; 77109c7e63aSJakub Grajciar int ret; 77209c7e63aSJakub Grajciar 77309c7e63aSJakub Grajciar ret = memif_msg_receive(cc); 77409c7e63aSJakub Grajciar /* if driver failed to assign device */ 77509c7e63aSJakub Grajciar if (cc->dev == NULL) { 77609c7e63aSJakub Grajciar ret = rte_intr_callback_unregister_pending(&cc->intr_handle, 77709c7e63aSJakub Grajciar memif_intr_handler, 77809c7e63aSJakub Grajciar cc, 77909c7e63aSJakub Grajciar memif_intr_unregister_handler); 78009c7e63aSJakub Grajciar if (ret < 0) 78109c7e63aSJakub Grajciar MIF_LOG(WARNING, 78209c7e63aSJakub Grajciar "Failed to unregister control channel callback."); 78309c7e63aSJakub Grajciar return; 78409c7e63aSJakub Grajciar } 78509c7e63aSJakub Grajciar /* if memif_msg_receive failed */ 78609c7e63aSJakub Grajciar if (ret < 0) 78709c7e63aSJakub Grajciar goto disconnect; 78809c7e63aSJakub Grajciar 78909c7e63aSJakub Grajciar ret = memif_msg_send_from_queue(cc); 79009c7e63aSJakub Grajciar if (ret < 0) 79109c7e63aSJakub Grajciar goto disconnect; 79209c7e63aSJakub Grajciar 79309c7e63aSJakub Grajciar return; 79409c7e63aSJakub Grajciar 79509c7e63aSJakub Grajciar disconnect: 79609c7e63aSJakub Grajciar if (cc->dev == NULL) { 79709c7e63aSJakub Grajciar MIF_LOG(WARNING, "eth dev not allocated"); 79809c7e63aSJakub Grajciar return; 79909c7e63aSJakub Grajciar } 80009c7e63aSJakub Grajciar memif_disconnect(cc->dev); 80109c7e63aSJakub Grajciar } 80209c7e63aSJakub Grajciar 80309c7e63aSJakub Grajciar static void 80409c7e63aSJakub Grajciar memif_listener_handler(void *arg) 80509c7e63aSJakub Grajciar { 80609c7e63aSJakub Grajciar struct memif_socket *socket = arg; 80709c7e63aSJakub Grajciar int sockfd; 80809c7e63aSJakub Grajciar int addr_len; 80909c7e63aSJakub Grajciar struct sockaddr_un client; 81009c7e63aSJakub Grajciar struct memif_control_channel *cc; 81109c7e63aSJakub Grajciar int ret; 81209c7e63aSJakub Grajciar 81309c7e63aSJakub Grajciar addr_len = sizeof(client); 81409c7e63aSJakub Grajciar sockfd = accept(socket->intr_handle.fd, (struct sockaddr *)&client, 81509c7e63aSJakub Grajciar (socklen_t *)&addr_len); 81609c7e63aSJakub Grajciar if (sockfd < 0) { 81709c7e63aSJakub Grajciar MIF_LOG(ERR, 81809c7e63aSJakub Grajciar "Failed to accept connection request on socket fd %d", 81909c7e63aSJakub Grajciar socket->intr_handle.fd); 82009c7e63aSJakub Grajciar return; 82109c7e63aSJakub Grajciar } 82209c7e63aSJakub Grajciar 82309c7e63aSJakub Grajciar MIF_LOG(DEBUG, "%s: Connection request accepted.", socket->filename); 82409c7e63aSJakub Grajciar 82509c7e63aSJakub Grajciar cc = rte_zmalloc("memif-cc", sizeof(struct memif_control_channel), 0); 82609c7e63aSJakub Grajciar if (cc == NULL) { 82709c7e63aSJakub Grajciar MIF_LOG(ERR, "Failed to allocate control channel."); 82809c7e63aSJakub Grajciar goto error; 82909c7e63aSJakub Grajciar } 83009c7e63aSJakub Grajciar 83109c7e63aSJakub Grajciar cc->intr_handle.fd = sockfd; 83209c7e63aSJakub Grajciar cc->intr_handle.type = RTE_INTR_HANDLE_EXT; 83309c7e63aSJakub Grajciar cc->socket = socket; 83409c7e63aSJakub Grajciar cc->dev = NULL; 83509c7e63aSJakub Grajciar TAILQ_INIT(&cc->msg_queue); 83609c7e63aSJakub Grajciar 83709c7e63aSJakub Grajciar ret = rte_intr_callback_register(&cc->intr_handle, memif_intr_handler, cc); 83809c7e63aSJakub Grajciar if (ret < 0) { 83909c7e63aSJakub Grajciar MIF_LOG(ERR, "Failed to register control channel callback."); 84009c7e63aSJakub Grajciar goto error; 84109c7e63aSJakub Grajciar } 84209c7e63aSJakub Grajciar 84309c7e63aSJakub Grajciar ret = memif_msg_enq_hello(cc); 84409c7e63aSJakub Grajciar if (ret < 0) { 84509c7e63aSJakub Grajciar MIF_LOG(ERR, "Failed to enqueue hello message."); 84609c7e63aSJakub Grajciar goto error; 84709c7e63aSJakub Grajciar } 84809c7e63aSJakub Grajciar ret = memif_msg_send_from_queue(cc); 84909c7e63aSJakub Grajciar if (ret < 0) 85009c7e63aSJakub Grajciar goto error; 85109c7e63aSJakub Grajciar 85209c7e63aSJakub Grajciar return; 85309c7e63aSJakub Grajciar 85409c7e63aSJakub Grajciar error: 855*10bd0ad5SJakub Grajciar if (sockfd >= 0) { 85609c7e63aSJakub Grajciar close(sockfd); 85709c7e63aSJakub Grajciar sockfd = -1; 85809c7e63aSJakub Grajciar } 85909c7e63aSJakub Grajciar if (cc != NULL) 86009c7e63aSJakub Grajciar rte_free(cc); 86109c7e63aSJakub Grajciar } 86209c7e63aSJakub Grajciar 86309c7e63aSJakub Grajciar static struct memif_socket * 86409c7e63aSJakub Grajciar memif_socket_create(struct pmd_internals *pmd, char *key, uint8_t listener) 86509c7e63aSJakub Grajciar { 86609c7e63aSJakub Grajciar struct memif_socket *sock; 86709c7e63aSJakub Grajciar struct sockaddr_un un; 86809c7e63aSJakub Grajciar int sockfd; 86909c7e63aSJakub Grajciar int ret; 87009c7e63aSJakub Grajciar int on = 1; 87109c7e63aSJakub Grajciar 87209c7e63aSJakub Grajciar sock = rte_zmalloc("memif-socket", sizeof(struct memif_socket), 0); 87309c7e63aSJakub Grajciar if (sock == NULL) { 87409c7e63aSJakub Grajciar MIF_LOG(ERR, "Failed to allocate memory for memif socket"); 87509c7e63aSJakub Grajciar return NULL; 87609c7e63aSJakub Grajciar } 87709c7e63aSJakub Grajciar 87809c7e63aSJakub Grajciar sock->listener = listener; 87909c7e63aSJakub Grajciar rte_memcpy(sock->filename, key, 256); 88009c7e63aSJakub Grajciar TAILQ_INIT(&sock->dev_queue); 88109c7e63aSJakub Grajciar 88209c7e63aSJakub Grajciar if (listener != 0) { 88309c7e63aSJakub Grajciar sockfd = socket(AF_UNIX, SOCK_SEQPACKET, 0); 88409c7e63aSJakub Grajciar if (sockfd < 0) 88509c7e63aSJakub Grajciar goto error; 88609c7e63aSJakub Grajciar 88709c7e63aSJakub Grajciar un.sun_family = AF_UNIX; 88809c7e63aSJakub Grajciar memcpy(un.sun_path, sock->filename, 88909c7e63aSJakub Grajciar sizeof(un.sun_path) - 1); 89009c7e63aSJakub Grajciar 89109c7e63aSJakub Grajciar ret = setsockopt(sockfd, SOL_SOCKET, SO_PASSCRED, &on, 89209c7e63aSJakub Grajciar sizeof(on)); 89309c7e63aSJakub Grajciar if (ret < 0) 89409c7e63aSJakub Grajciar goto error; 89509c7e63aSJakub Grajciar ret = bind(sockfd, (struct sockaddr *)&un, sizeof(un)); 89609c7e63aSJakub Grajciar if (ret < 0) 89709c7e63aSJakub Grajciar goto error; 89809c7e63aSJakub Grajciar ret = listen(sockfd, 1); 89909c7e63aSJakub Grajciar if (ret < 0) 90009c7e63aSJakub Grajciar goto error; 90109c7e63aSJakub Grajciar 90209c7e63aSJakub Grajciar MIF_LOG(DEBUG, "%s: Memif listener socket %s created.", 90309c7e63aSJakub Grajciar rte_vdev_device_name(pmd->vdev), sock->filename); 90409c7e63aSJakub Grajciar 90509c7e63aSJakub Grajciar sock->intr_handle.fd = sockfd; 90609c7e63aSJakub Grajciar sock->intr_handle.type = RTE_INTR_HANDLE_EXT; 90709c7e63aSJakub Grajciar ret = rte_intr_callback_register(&sock->intr_handle, 90809c7e63aSJakub Grajciar memif_listener_handler, sock); 90909c7e63aSJakub Grajciar if (ret < 0) { 91009c7e63aSJakub Grajciar MIF_LOG(ERR, "%s: Failed to register interrupt " 91109c7e63aSJakub Grajciar "callback for listener socket", 91209c7e63aSJakub Grajciar rte_vdev_device_name(pmd->vdev)); 91309c7e63aSJakub Grajciar return NULL; 91409c7e63aSJakub Grajciar } 91509c7e63aSJakub Grajciar } 91609c7e63aSJakub Grajciar 91709c7e63aSJakub Grajciar return sock; 91809c7e63aSJakub Grajciar 91909c7e63aSJakub Grajciar error: 92009c7e63aSJakub Grajciar MIF_LOG(ERR, "%s: Failed to setup socket %s: %s", 92109c7e63aSJakub Grajciar rte_vdev_device_name(pmd->vdev), key, strerror(errno)); 92209c7e63aSJakub Grajciar if (sock != NULL) 92309c7e63aSJakub Grajciar rte_free(sock); 924*10bd0ad5SJakub Grajciar if (sockfd >= 0) 925*10bd0ad5SJakub Grajciar close(sockfd); 92609c7e63aSJakub Grajciar return NULL; 92709c7e63aSJakub Grajciar } 92809c7e63aSJakub Grajciar 92909c7e63aSJakub Grajciar static struct rte_hash * 93009c7e63aSJakub Grajciar memif_create_socket_hash(void) 93109c7e63aSJakub Grajciar { 93209c7e63aSJakub Grajciar struct rte_hash_parameters params = { 0 }; 93309c7e63aSJakub Grajciar params.name = MEMIF_SOCKET_HASH_NAME; 93409c7e63aSJakub Grajciar params.entries = 256; 93509c7e63aSJakub Grajciar params.key_len = 256; 93609c7e63aSJakub Grajciar params.hash_func = rte_jhash; 93709c7e63aSJakub Grajciar params.hash_func_init_val = 0; 93809c7e63aSJakub Grajciar return rte_hash_create(¶ms); 93909c7e63aSJakub Grajciar } 94009c7e63aSJakub Grajciar 94109c7e63aSJakub Grajciar int 94209c7e63aSJakub Grajciar memif_socket_init(struct rte_eth_dev *dev, const char *socket_filename) 94309c7e63aSJakub Grajciar { 94409c7e63aSJakub Grajciar struct pmd_internals *pmd = dev->data->dev_private; 94509c7e63aSJakub Grajciar struct memif_socket *socket = NULL; 94609c7e63aSJakub Grajciar struct memif_socket_dev_list_elt *elt; 94709c7e63aSJakub Grajciar struct pmd_internals *tmp_pmd; 94809c7e63aSJakub Grajciar struct rte_hash *hash; 94909c7e63aSJakub Grajciar int ret; 95009c7e63aSJakub Grajciar char key[256]; 95109c7e63aSJakub Grajciar 95209c7e63aSJakub Grajciar hash = rte_hash_find_existing(MEMIF_SOCKET_HASH_NAME); 95309c7e63aSJakub Grajciar if (hash == NULL) { 95409c7e63aSJakub Grajciar hash = memif_create_socket_hash(); 95509c7e63aSJakub Grajciar if (hash == NULL) { 95609c7e63aSJakub Grajciar MIF_LOG(ERR, "Failed to create memif socket hash."); 95709c7e63aSJakub Grajciar return -1; 95809c7e63aSJakub Grajciar } 95909c7e63aSJakub Grajciar } 96009c7e63aSJakub Grajciar 96109c7e63aSJakub Grajciar memset(key, 0, 256); 96209c7e63aSJakub Grajciar rte_memcpy(key, socket_filename, strlen(socket_filename)); 96309c7e63aSJakub Grajciar ret = rte_hash_lookup_data(hash, key, (void **)&socket); 96409c7e63aSJakub Grajciar if (ret < 0) { 96509c7e63aSJakub Grajciar socket = memif_socket_create(pmd, key, 96609c7e63aSJakub Grajciar (pmd->role == 96709c7e63aSJakub Grajciar MEMIF_ROLE_SLAVE) ? 0 : 1); 96809c7e63aSJakub Grajciar if (socket == NULL) 96909c7e63aSJakub Grajciar return -1; 97009c7e63aSJakub Grajciar ret = rte_hash_add_key_data(hash, key, socket); 97109c7e63aSJakub Grajciar if (ret < 0) { 97209c7e63aSJakub Grajciar MIF_LOG(ERR, "Failed to add socket to socket hash."); 97309c7e63aSJakub Grajciar return ret; 97409c7e63aSJakub Grajciar } 97509c7e63aSJakub Grajciar } 97609c7e63aSJakub Grajciar pmd->socket_filename = socket->filename; 97709c7e63aSJakub Grajciar 97809c7e63aSJakub Grajciar if (socket->listener != 0 && pmd->role == MEMIF_ROLE_SLAVE) { 97909c7e63aSJakub Grajciar MIF_LOG(ERR, "Socket is a listener."); 98009c7e63aSJakub Grajciar return -1; 98109c7e63aSJakub Grajciar } else if ((socket->listener == 0) && (pmd->role == MEMIF_ROLE_MASTER)) { 98209c7e63aSJakub Grajciar MIF_LOG(ERR, "Socket is not a listener."); 98309c7e63aSJakub Grajciar return -1; 98409c7e63aSJakub Grajciar } 98509c7e63aSJakub Grajciar 98609c7e63aSJakub Grajciar TAILQ_FOREACH(elt, &socket->dev_queue, next) { 98709c7e63aSJakub Grajciar tmp_pmd = elt->dev->data->dev_private; 98809c7e63aSJakub Grajciar if (tmp_pmd->id == pmd->id) { 98909c7e63aSJakub Grajciar MIF_LOG(ERR, "Memif device with id %d already " 99009c7e63aSJakub Grajciar "exists on socket %s", 99109c7e63aSJakub Grajciar pmd->id, socket->filename); 99209c7e63aSJakub Grajciar return -1; 99309c7e63aSJakub Grajciar } 99409c7e63aSJakub Grajciar } 99509c7e63aSJakub Grajciar 99609c7e63aSJakub Grajciar elt = rte_malloc("pmd-queue", sizeof(struct memif_socket_dev_list_elt), 0); 99709c7e63aSJakub Grajciar if (elt == NULL) { 99809c7e63aSJakub Grajciar MIF_LOG(ERR, "%s: Failed to add device to socket device list.", 99909c7e63aSJakub Grajciar rte_vdev_device_name(pmd->vdev)); 100009c7e63aSJakub Grajciar return -1; 100109c7e63aSJakub Grajciar } 100209c7e63aSJakub Grajciar elt->dev = dev; 100309c7e63aSJakub Grajciar TAILQ_INSERT_TAIL(&socket->dev_queue, elt, next); 100409c7e63aSJakub Grajciar 100509c7e63aSJakub Grajciar return 0; 100609c7e63aSJakub Grajciar } 100709c7e63aSJakub Grajciar 100809c7e63aSJakub Grajciar void 100909c7e63aSJakub Grajciar memif_socket_remove_device(struct rte_eth_dev *dev) 101009c7e63aSJakub Grajciar { 101109c7e63aSJakub Grajciar struct pmd_internals *pmd = dev->data->dev_private; 101209c7e63aSJakub Grajciar struct memif_socket *socket = NULL; 101309c7e63aSJakub Grajciar struct memif_socket_dev_list_elt *elt, *next; 101409c7e63aSJakub Grajciar struct rte_hash *hash; 1015*10bd0ad5SJakub Grajciar int ret; 101609c7e63aSJakub Grajciar 101709c7e63aSJakub Grajciar hash = rte_hash_find_existing(MEMIF_SOCKET_HASH_NAME); 101809c7e63aSJakub Grajciar if (hash == NULL) 101909c7e63aSJakub Grajciar return; 102009c7e63aSJakub Grajciar 102109c7e63aSJakub Grajciar if (pmd->socket_filename == NULL) 102209c7e63aSJakub Grajciar return; 102309c7e63aSJakub Grajciar 102409c7e63aSJakub Grajciar if (rte_hash_lookup_data(hash, pmd->socket_filename, (void **)&socket) < 0) 102509c7e63aSJakub Grajciar return; 102609c7e63aSJakub Grajciar 102709c7e63aSJakub Grajciar for (elt = TAILQ_FIRST(&socket->dev_queue); elt != NULL; elt = next) { 102809c7e63aSJakub Grajciar next = TAILQ_NEXT(elt, next); 102909c7e63aSJakub Grajciar if (elt->dev == dev) { 103009c7e63aSJakub Grajciar TAILQ_REMOVE(&socket->dev_queue, elt, next); 103109c7e63aSJakub Grajciar rte_free(elt); 103209c7e63aSJakub Grajciar pmd->socket_filename = NULL; 103309c7e63aSJakub Grajciar } 103409c7e63aSJakub Grajciar } 103509c7e63aSJakub Grajciar 103609c7e63aSJakub Grajciar /* remove socket, if this was the last device using it */ 103709c7e63aSJakub Grajciar if (TAILQ_EMPTY(&socket->dev_queue)) { 103809c7e63aSJakub Grajciar rte_hash_del_key(hash, socket->filename); 103909c7e63aSJakub Grajciar if (socket->listener) { 104009c7e63aSJakub Grajciar /* remove listener socket file, 104109c7e63aSJakub Grajciar * so we can create new one later. 104209c7e63aSJakub Grajciar */ 1043*10bd0ad5SJakub Grajciar ret = remove(socket->filename); 1044*10bd0ad5SJakub Grajciar if (ret < 0) 1045*10bd0ad5SJakub Grajciar MIF_LOG(ERR, "Failed to remove socket file: %s", 1046*10bd0ad5SJakub Grajciar socket->filename); 104709c7e63aSJakub Grajciar } 104809c7e63aSJakub Grajciar rte_free(socket); 104909c7e63aSJakub Grajciar } 105009c7e63aSJakub Grajciar } 105109c7e63aSJakub Grajciar 105209c7e63aSJakub Grajciar int 105309c7e63aSJakub Grajciar memif_connect_master(struct rte_eth_dev *dev) 105409c7e63aSJakub Grajciar { 105509c7e63aSJakub Grajciar struct pmd_internals *pmd = dev->data->dev_private; 105609c7e63aSJakub Grajciar 105709c7e63aSJakub Grajciar memset(pmd->local_disc_string, 0, ETH_MEMIF_DISC_STRING_SIZE); 105809c7e63aSJakub Grajciar memset(pmd->remote_disc_string, 0, ETH_MEMIF_DISC_STRING_SIZE); 105909c7e63aSJakub Grajciar pmd->flags &= ~ETH_MEMIF_FLAG_DISABLED; 106009c7e63aSJakub Grajciar return 0; 106109c7e63aSJakub Grajciar } 106209c7e63aSJakub Grajciar 106309c7e63aSJakub Grajciar int 106409c7e63aSJakub Grajciar memif_connect_slave(struct rte_eth_dev *dev) 106509c7e63aSJakub Grajciar { 106609c7e63aSJakub Grajciar int sockfd; 106709c7e63aSJakub Grajciar int ret; 106809c7e63aSJakub Grajciar struct sockaddr_un sun; 106909c7e63aSJakub Grajciar struct pmd_internals *pmd = dev->data->dev_private; 107009c7e63aSJakub Grajciar 107109c7e63aSJakub Grajciar memset(pmd->local_disc_string, 0, ETH_MEMIF_DISC_STRING_SIZE); 107209c7e63aSJakub Grajciar memset(pmd->remote_disc_string, 0, ETH_MEMIF_DISC_STRING_SIZE); 107309c7e63aSJakub Grajciar pmd->flags &= ~ETH_MEMIF_FLAG_DISABLED; 107409c7e63aSJakub Grajciar 107509c7e63aSJakub Grajciar sockfd = socket(AF_UNIX, SOCK_SEQPACKET, 0); 107609c7e63aSJakub Grajciar if (sockfd < 0) { 107709c7e63aSJakub Grajciar MIF_LOG(ERR, "%s: Failed to open socket.", 107809c7e63aSJakub Grajciar rte_vdev_device_name(pmd->vdev)); 107909c7e63aSJakub Grajciar return -1; 108009c7e63aSJakub Grajciar } 108109c7e63aSJakub Grajciar 108209c7e63aSJakub Grajciar sun.sun_family = AF_UNIX; 108309c7e63aSJakub Grajciar 108409c7e63aSJakub Grajciar memcpy(sun.sun_path, pmd->socket_filename, sizeof(sun.sun_path) - 1); 108509c7e63aSJakub Grajciar 108609c7e63aSJakub Grajciar ret = connect(sockfd, (struct sockaddr *)&sun, 108709c7e63aSJakub Grajciar sizeof(struct sockaddr_un)); 108809c7e63aSJakub Grajciar if (ret < 0) { 108909c7e63aSJakub Grajciar MIF_LOG(ERR, "%s: Failed to connect socket: %s.", 109009c7e63aSJakub Grajciar rte_vdev_device_name(pmd->vdev), pmd->socket_filename); 109109c7e63aSJakub Grajciar goto error; 109209c7e63aSJakub Grajciar } 109309c7e63aSJakub Grajciar 109409c7e63aSJakub Grajciar MIF_LOG(DEBUG, "%s: Memif socket: %s connected.", 109509c7e63aSJakub Grajciar rte_vdev_device_name(pmd->vdev), pmd->socket_filename); 109609c7e63aSJakub Grajciar 109709c7e63aSJakub Grajciar pmd->cc = rte_zmalloc("memif-cc", 109809c7e63aSJakub Grajciar sizeof(struct memif_control_channel), 0); 109909c7e63aSJakub Grajciar if (pmd->cc == NULL) { 110009c7e63aSJakub Grajciar MIF_LOG(ERR, "%s: Failed to allocate control channel.", 110109c7e63aSJakub Grajciar rte_vdev_device_name(pmd->vdev)); 110209c7e63aSJakub Grajciar goto error; 110309c7e63aSJakub Grajciar } 110409c7e63aSJakub Grajciar 110509c7e63aSJakub Grajciar pmd->cc->intr_handle.fd = sockfd; 110609c7e63aSJakub Grajciar pmd->cc->intr_handle.type = RTE_INTR_HANDLE_EXT; 110709c7e63aSJakub Grajciar pmd->cc->socket = NULL; 110809c7e63aSJakub Grajciar pmd->cc->dev = dev; 110909c7e63aSJakub Grajciar TAILQ_INIT(&pmd->cc->msg_queue); 111009c7e63aSJakub Grajciar 111109c7e63aSJakub Grajciar ret = rte_intr_callback_register(&pmd->cc->intr_handle, 111209c7e63aSJakub Grajciar memif_intr_handler, pmd->cc); 111309c7e63aSJakub Grajciar if (ret < 0) { 111409c7e63aSJakub Grajciar MIF_LOG(ERR, "%s: Failed to register interrupt callback " 111509c7e63aSJakub Grajciar "for control fd", rte_vdev_device_name(pmd->vdev)); 111609c7e63aSJakub Grajciar goto error; 111709c7e63aSJakub Grajciar } 111809c7e63aSJakub Grajciar 111909c7e63aSJakub Grajciar return 0; 112009c7e63aSJakub Grajciar 112109c7e63aSJakub Grajciar error: 1122*10bd0ad5SJakub Grajciar if (sockfd >= 0) { 112309c7e63aSJakub Grajciar close(sockfd); 112409c7e63aSJakub Grajciar sockfd = -1; 112509c7e63aSJakub Grajciar } 112609c7e63aSJakub Grajciar if (pmd->cc != NULL) { 112709c7e63aSJakub Grajciar rte_free(pmd->cc); 112809c7e63aSJakub Grajciar pmd->cc = NULL; 112909c7e63aSJakub Grajciar } 113009c7e63aSJakub Grajciar return -1; 113109c7e63aSJakub Grajciar } 1132