18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
28c2ecf20Sopenharmony_ci/*
38c2ecf20Sopenharmony_ci * Copyright (C) 2001 Lennert Buytenhek (buytenh@gnu.org) and
48c2ecf20Sopenharmony_ci * James Leu (jleu@mindspring.net).
58c2ecf20Sopenharmony_ci * Copyright (C) 2001 - 2007 Jeff Dike (jdike@{addtoit,linux.intel}.com)
68c2ecf20Sopenharmony_ci * Copyright (C) 2001 by various other people who didn't put their name here.
78c2ecf20Sopenharmony_ci */
88c2ecf20Sopenharmony_ci
98c2ecf20Sopenharmony_ci#include <linux/init.h>
108c2ecf20Sopenharmony_ci#include <linux/netdevice.h>
118c2ecf20Sopenharmony_ci#include "etap.h"
128c2ecf20Sopenharmony_ci#include <net_kern.h>
138c2ecf20Sopenharmony_ci
148c2ecf20Sopenharmony_cistruct ethertap_init {
158c2ecf20Sopenharmony_ci	char *dev_name;
168c2ecf20Sopenharmony_ci	char *gate_addr;
178c2ecf20Sopenharmony_ci};
188c2ecf20Sopenharmony_ci
198c2ecf20Sopenharmony_cistatic void etap_init(struct net_device *dev, void *data)
208c2ecf20Sopenharmony_ci{
218c2ecf20Sopenharmony_ci	struct uml_net_private *pri;
228c2ecf20Sopenharmony_ci	struct ethertap_data *epri;
238c2ecf20Sopenharmony_ci	struct ethertap_init *init = data;
248c2ecf20Sopenharmony_ci
258c2ecf20Sopenharmony_ci	pri = netdev_priv(dev);
268c2ecf20Sopenharmony_ci	epri = (struct ethertap_data *) pri->user;
278c2ecf20Sopenharmony_ci	epri->dev_name = init->dev_name;
288c2ecf20Sopenharmony_ci	epri->gate_addr = init->gate_addr;
298c2ecf20Sopenharmony_ci	epri->data_fd = -1;
308c2ecf20Sopenharmony_ci	epri->control_fd = -1;
318c2ecf20Sopenharmony_ci	epri->dev = dev;
328c2ecf20Sopenharmony_ci
338c2ecf20Sopenharmony_ci	printk(KERN_INFO "ethertap backend - %s", epri->dev_name);
348c2ecf20Sopenharmony_ci	if (epri->gate_addr != NULL)
358c2ecf20Sopenharmony_ci		printk(KERN_CONT ", IP = %s", epri->gate_addr);
368c2ecf20Sopenharmony_ci	printk(KERN_CONT "\n");
378c2ecf20Sopenharmony_ci}
388c2ecf20Sopenharmony_ci
398c2ecf20Sopenharmony_cistatic int etap_read(int fd, struct sk_buff *skb, struct uml_net_private *lp)
408c2ecf20Sopenharmony_ci{
418c2ecf20Sopenharmony_ci	int len;
428c2ecf20Sopenharmony_ci
438c2ecf20Sopenharmony_ci	len = net_recvfrom(fd, skb_mac_header(skb),
448c2ecf20Sopenharmony_ci			   skb->dev->mtu + 2 + ETH_HEADER_ETHERTAP);
458c2ecf20Sopenharmony_ci	if (len <= 0)
468c2ecf20Sopenharmony_ci		return(len);
478c2ecf20Sopenharmony_ci
488c2ecf20Sopenharmony_ci	skb_pull(skb, 2);
498c2ecf20Sopenharmony_ci	len -= 2;
508c2ecf20Sopenharmony_ci	return len;
518c2ecf20Sopenharmony_ci}
528c2ecf20Sopenharmony_ci
538c2ecf20Sopenharmony_cistatic int etap_write(int fd, struct sk_buff *skb, struct uml_net_private *lp)
548c2ecf20Sopenharmony_ci{
558c2ecf20Sopenharmony_ci	skb_push(skb, 2);
568c2ecf20Sopenharmony_ci	return net_send(fd, skb->data, skb->len);
578c2ecf20Sopenharmony_ci}
588c2ecf20Sopenharmony_ci
598c2ecf20Sopenharmony_ciconst struct net_kern_info ethertap_kern_info = {
608c2ecf20Sopenharmony_ci	.init			= etap_init,
618c2ecf20Sopenharmony_ci	.protocol		= eth_protocol,
628c2ecf20Sopenharmony_ci	.read			= etap_read,
638c2ecf20Sopenharmony_ci	.write 			= etap_write,
648c2ecf20Sopenharmony_ci};
658c2ecf20Sopenharmony_ci
668c2ecf20Sopenharmony_ciint ethertap_setup(char *str, char **mac_out, void *data)
678c2ecf20Sopenharmony_ci{
688c2ecf20Sopenharmony_ci	struct ethertap_init *init = data;
698c2ecf20Sopenharmony_ci
708c2ecf20Sopenharmony_ci	*init = ((struct ethertap_init)
718c2ecf20Sopenharmony_ci		{ .dev_name 	= NULL,
728c2ecf20Sopenharmony_ci		  .gate_addr 	= NULL });
738c2ecf20Sopenharmony_ci	if (tap_setup_common(str, "ethertap", &init->dev_name, mac_out,
748c2ecf20Sopenharmony_ci			    &init->gate_addr))
758c2ecf20Sopenharmony_ci		return 0;
768c2ecf20Sopenharmony_ci	if (init->dev_name == NULL) {
778c2ecf20Sopenharmony_ci		printk(KERN_ERR "ethertap_setup : Missing tap device name\n");
788c2ecf20Sopenharmony_ci		return 0;
798c2ecf20Sopenharmony_ci	}
808c2ecf20Sopenharmony_ci
818c2ecf20Sopenharmony_ci	return 1;
828c2ecf20Sopenharmony_ci}
838c2ecf20Sopenharmony_ci
848c2ecf20Sopenharmony_cistatic struct transport ethertap_transport = {
858c2ecf20Sopenharmony_ci	.list 		= LIST_HEAD_INIT(ethertap_transport.list),
868c2ecf20Sopenharmony_ci	.name 		= "ethertap",
878c2ecf20Sopenharmony_ci	.setup  	= ethertap_setup,
888c2ecf20Sopenharmony_ci	.user 		= &ethertap_user_info,
898c2ecf20Sopenharmony_ci	.kern 		= &ethertap_kern_info,
908c2ecf20Sopenharmony_ci	.private_size 	= sizeof(struct ethertap_data),
918c2ecf20Sopenharmony_ci	.setup_size 	= sizeof(struct ethertap_init),
928c2ecf20Sopenharmony_ci};
938c2ecf20Sopenharmony_ci
948c2ecf20Sopenharmony_cistatic int register_ethertap(void)
958c2ecf20Sopenharmony_ci{
968c2ecf20Sopenharmony_ci	register_transport(&ethertap_transport);
978c2ecf20Sopenharmony_ci	return 0;
988c2ecf20Sopenharmony_ci}
998c2ecf20Sopenharmony_ci
1008c2ecf20Sopenharmony_cilate_initcall(register_ethertap);
101