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