xref: /openbmc/linux/arch/um/drivers/vde_user.c (revision 976e3645923bdd2fe7893aae33fd7a21098bfb28)
1*dbddf429SAlex Dewar // SPDX-License-Identifier: GPL-2.0
2ad43c356SJeff Dike /*
3ad43c356SJeff Dike  * Copyright (C) 2007 Luca Bigliardi (shammash@artha.org).
4ad43c356SJeff Dike  */
5ad43c356SJeff Dike 
6cd1ae0e4SJeff Dike #include <stddef.h>
7ad43c356SJeff Dike #include <errno.h>
8ad43c356SJeff Dike #include <libvdeplug.h>
937185b33SAl Viro #include <net_user.h>
1037185b33SAl Viro #include <um_malloc.h>
11ad43c356SJeff Dike #include "vde.h"
12ad43c356SJeff Dike 
vde_user_init(void * data,void * dev)13ad43c356SJeff Dike static int vde_user_init(void *data, void *dev)
14ad43c356SJeff Dike {
15ad43c356SJeff Dike 	struct vde_data *pri = data;
16ad43c356SJeff Dike 	VDECONN *conn = NULL;
17ad43c356SJeff Dike 	int err = -EINVAL;
18ad43c356SJeff Dike 
19ad43c356SJeff Dike 	pri->dev = dev;
20ad43c356SJeff Dike 
21ad43c356SJeff Dike 	conn = vde_open(pri->vde_switch, pri->descr, pri->args);
22ad43c356SJeff Dike 
23ad43c356SJeff Dike 	if (conn == NULL) {
24ad43c356SJeff Dike 		err = -errno;
25ad43c356SJeff Dike 		printk(UM_KERN_ERR "vde_user_init: vde_open failed, "
26ad43c356SJeff Dike 		       "errno = %d\n", errno);
27ad43c356SJeff Dike 		return err;
28ad43c356SJeff Dike 	}
29ad43c356SJeff Dike 
30ad43c356SJeff Dike 	printk(UM_KERN_INFO "vde backend - connection opened\n");
31ad43c356SJeff Dike 
32ad43c356SJeff Dike 	pri->conn = conn;
33ad43c356SJeff Dike 
34ad43c356SJeff Dike 	return 0;
35ad43c356SJeff Dike }
36ad43c356SJeff Dike 
vde_user_open(void * data)37ad43c356SJeff Dike static int vde_user_open(void *data)
38ad43c356SJeff Dike {
39ad43c356SJeff Dike 	struct vde_data *pri = data;
40ad43c356SJeff Dike 
41ad43c356SJeff Dike 	if (pri->conn != NULL)
42ad43c356SJeff Dike 		return vde_datafd(pri->conn);
43ad43c356SJeff Dike 
44ad43c356SJeff Dike 	printk(UM_KERN_WARNING "vde_open - we have no VDECONN to open");
45ad43c356SJeff Dike 	return -EINVAL;
46ad43c356SJeff Dike }
47ad43c356SJeff Dike 
vde_remove(void * data)48ad43c356SJeff Dike static void vde_remove(void *data)
49ad43c356SJeff Dike {
50ad43c356SJeff Dike 	struct vde_data *pri = data;
51ad43c356SJeff Dike 
52ad43c356SJeff Dike 	if (pri->conn != NULL) {
53ad43c356SJeff Dike 		printk(UM_KERN_INFO "vde backend - closing connection\n");
54ad43c356SJeff Dike 		vde_close(pri->conn);
55ad43c356SJeff Dike 		pri->conn = NULL;
56ad43c356SJeff Dike 		kfree(pri->args);
57ad43c356SJeff Dike 		pri->args = NULL;
58ad43c356SJeff Dike 		return;
59ad43c356SJeff Dike 	}
60ad43c356SJeff Dike 
61ad43c356SJeff Dike 	printk(UM_KERN_WARNING "vde_remove - we have no VDECONN to remove");
62ad43c356SJeff Dike }
63ad43c356SJeff Dike 
64ad43c356SJeff Dike const struct net_user_info vde_user_info = {
65ad43c356SJeff Dike 	.init		= vde_user_init,
66ad43c356SJeff Dike 	.open		= vde_user_open,
67ad43c356SJeff Dike 	.close	 	= NULL,
68ad43c356SJeff Dike 	.remove	 	= vde_remove,
69ad43c356SJeff Dike 	.add_address	= NULL,
70ad43c356SJeff Dike 	.delete_address = NULL,
71b53f35a8SJeff Dike 	.mtu		= ETH_MAX_PACKET,
72b53f35a8SJeff Dike 	.max_packet	= ETH_MAX_PACKET + ETH_HEADER_OTHER,
73ad43c356SJeff Dike };
74ad43c356SJeff Dike 
vde_init_libstuff(struct vde_data * vpri,struct vde_init * init)75ad43c356SJeff Dike void vde_init_libstuff(struct vde_data *vpri, struct vde_init *init)
76ad43c356SJeff Dike {
77ad43c356SJeff Dike 	struct vde_open_args *args;
78ad43c356SJeff Dike 
7941a9e64cSLuca Bigliardi 	vpri->args = uml_kmalloc(sizeof(struct vde_open_args), UM_GFP_KERNEL);
80ad43c356SJeff Dike 	if (vpri->args == NULL) {
81ad43c356SJeff Dike 		printk(UM_KERN_ERR "vde_init_libstuff - vde_open_args "
82ad43c356SJeff Dike 		       "allocation failed");
83ad43c356SJeff Dike 		return;
84ad43c356SJeff Dike 	}
85ad43c356SJeff Dike 
86ad43c356SJeff Dike 	args = vpri->args;
87ad43c356SJeff Dike 
88ad43c356SJeff Dike 	args->port = init->port;
89ad43c356SJeff Dike 	args->group = init->group;
90ad43c356SJeff Dike 	args->mode = init->mode ? init->mode : 0700;
91ad43c356SJeff Dike 
9241a9e64cSLuca Bigliardi 	args->port ?  printk("port %d", args->port) :
9341a9e64cSLuca Bigliardi 		printk("undefined port");
94ad43c356SJeff Dike }
95ad43c356SJeff Dike 
vde_user_read(void * conn,void * buf,int len)96ad43c356SJeff Dike int vde_user_read(void *conn, void *buf, int len)
97ad43c356SJeff Dike {
98ad43c356SJeff Dike 	VDECONN *vconn = conn;
99ad43c356SJeff Dike 	int rv;
100ad43c356SJeff Dike 
101ad43c356SJeff Dike 	if (vconn == NULL)
102ad43c356SJeff Dike 		return 0;
103ad43c356SJeff Dike 
104ad43c356SJeff Dike 	rv = vde_recv(vconn, buf, len, 0);
105ad43c356SJeff Dike 	if (rv < 0) {
106ad43c356SJeff Dike 		if (errno == EAGAIN)
107ad43c356SJeff Dike 			return 0;
108ad43c356SJeff Dike 		return -errno;
109ad43c356SJeff Dike 	}
110ad43c356SJeff Dike 	else if (rv == 0)
111ad43c356SJeff Dike 		return -ENOTCONN;
112ad43c356SJeff Dike 
113ad43c356SJeff Dike 	return rv;
114ad43c356SJeff Dike }
115ad43c356SJeff Dike 
vde_user_write(void * conn,void * buf,int len)116ad43c356SJeff Dike int vde_user_write(void *conn, void *buf, int len)
117ad43c356SJeff Dike {
118ad43c356SJeff Dike 	VDECONN *vconn = conn;
119ad43c356SJeff Dike 
120ad43c356SJeff Dike 	if (vconn == NULL)
121ad43c356SJeff Dike 		return 0;
122ad43c356SJeff Dike 
123ad43c356SJeff Dike 	return vde_send(vconn, buf, len, 0);
124ad43c356SJeff Dike }
125ad43c356SJeff Dike 
126