162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * Copyright (C) 2001 Lennert Buytenhek (buytenh@gnu.org) and
462306a36Sopenharmony_ci * James Leu (jleu@mindspring.net).
562306a36Sopenharmony_ci * Copyright (C) 2001 - 2007 Jeff Dike (jdike@{addtoit,linux.intel}.com)
662306a36Sopenharmony_ci * Copyright (C) 2001 by various other people who didn't put their name here.
762306a36Sopenharmony_ci */
862306a36Sopenharmony_ci
962306a36Sopenharmony_ci#include <linux/init.h>
1062306a36Sopenharmony_ci#include <linux/netdevice.h>
1162306a36Sopenharmony_ci#include "etap.h"
1262306a36Sopenharmony_ci#include <net_kern.h>
1362306a36Sopenharmony_ci
1462306a36Sopenharmony_cistruct ethertap_init {
1562306a36Sopenharmony_ci	char *dev_name;
1662306a36Sopenharmony_ci	char *gate_addr;
1762306a36Sopenharmony_ci};
1862306a36Sopenharmony_ci
1962306a36Sopenharmony_cistatic void etap_init(struct net_device *dev, void *data)
2062306a36Sopenharmony_ci{
2162306a36Sopenharmony_ci	struct uml_net_private *pri;
2262306a36Sopenharmony_ci	struct ethertap_data *epri;
2362306a36Sopenharmony_ci	struct ethertap_init *init = data;
2462306a36Sopenharmony_ci
2562306a36Sopenharmony_ci	pri = netdev_priv(dev);
2662306a36Sopenharmony_ci	epri = (struct ethertap_data *) pri->user;
2762306a36Sopenharmony_ci	epri->dev_name = init->dev_name;
2862306a36Sopenharmony_ci	epri->gate_addr = init->gate_addr;
2962306a36Sopenharmony_ci	epri->data_fd = -1;
3062306a36Sopenharmony_ci	epri->control_fd = -1;
3162306a36Sopenharmony_ci	epri->dev = dev;
3262306a36Sopenharmony_ci
3362306a36Sopenharmony_ci	printk(KERN_INFO "ethertap backend - %s", epri->dev_name);
3462306a36Sopenharmony_ci	if (epri->gate_addr != NULL)
3562306a36Sopenharmony_ci		printk(KERN_CONT ", IP = %s", epri->gate_addr);
3662306a36Sopenharmony_ci	printk(KERN_CONT "\n");
3762306a36Sopenharmony_ci}
3862306a36Sopenharmony_ci
3962306a36Sopenharmony_cistatic int etap_read(int fd, struct sk_buff *skb, struct uml_net_private *lp)
4062306a36Sopenharmony_ci{
4162306a36Sopenharmony_ci	int len;
4262306a36Sopenharmony_ci
4362306a36Sopenharmony_ci	len = net_recvfrom(fd, skb_mac_header(skb),
4462306a36Sopenharmony_ci			   skb->dev->mtu + 2 + ETH_HEADER_ETHERTAP);
4562306a36Sopenharmony_ci	if (len <= 0)
4662306a36Sopenharmony_ci		return(len);
4762306a36Sopenharmony_ci
4862306a36Sopenharmony_ci	skb_pull(skb, 2);
4962306a36Sopenharmony_ci	len -= 2;
5062306a36Sopenharmony_ci	return len;
5162306a36Sopenharmony_ci}
5262306a36Sopenharmony_ci
5362306a36Sopenharmony_cistatic int etap_write(int fd, struct sk_buff *skb, struct uml_net_private *lp)
5462306a36Sopenharmony_ci{
5562306a36Sopenharmony_ci	skb_push(skb, 2);
5662306a36Sopenharmony_ci	return net_send(fd, skb->data, skb->len);
5762306a36Sopenharmony_ci}
5862306a36Sopenharmony_ci
5962306a36Sopenharmony_ciconst struct net_kern_info ethertap_kern_info = {
6062306a36Sopenharmony_ci	.init			= etap_init,
6162306a36Sopenharmony_ci	.protocol		= eth_protocol,
6262306a36Sopenharmony_ci	.read			= etap_read,
6362306a36Sopenharmony_ci	.write 			= etap_write,
6462306a36Sopenharmony_ci};
6562306a36Sopenharmony_ci
6662306a36Sopenharmony_ciint ethertap_setup(char *str, char **mac_out, void *data)
6762306a36Sopenharmony_ci{
6862306a36Sopenharmony_ci	struct ethertap_init *init = data;
6962306a36Sopenharmony_ci
7062306a36Sopenharmony_ci	*init = ((struct ethertap_init)
7162306a36Sopenharmony_ci		{ .dev_name 	= NULL,
7262306a36Sopenharmony_ci		  .gate_addr 	= NULL });
7362306a36Sopenharmony_ci	if (tap_setup_common(str, "ethertap", &init->dev_name, mac_out,
7462306a36Sopenharmony_ci			    &init->gate_addr))
7562306a36Sopenharmony_ci		return 0;
7662306a36Sopenharmony_ci	if (init->dev_name == NULL) {
7762306a36Sopenharmony_ci		printk(KERN_ERR "ethertap_setup : Missing tap device name\n");
7862306a36Sopenharmony_ci		return 0;
7962306a36Sopenharmony_ci	}
8062306a36Sopenharmony_ci
8162306a36Sopenharmony_ci	return 1;
8262306a36Sopenharmony_ci}
8362306a36Sopenharmony_ci
8462306a36Sopenharmony_cistatic struct transport ethertap_transport = {
8562306a36Sopenharmony_ci	.list 		= LIST_HEAD_INIT(ethertap_transport.list),
8662306a36Sopenharmony_ci	.name 		= "ethertap",
8762306a36Sopenharmony_ci	.setup  	= ethertap_setup,
8862306a36Sopenharmony_ci	.user 		= &ethertap_user_info,
8962306a36Sopenharmony_ci	.kern 		= &ethertap_kern_info,
9062306a36Sopenharmony_ci	.private_size 	= sizeof(struct ethertap_data),
9162306a36Sopenharmony_ci	.setup_size 	= sizeof(struct ethertap_init),
9262306a36Sopenharmony_ci};
9362306a36Sopenharmony_ci
9462306a36Sopenharmony_cistatic int register_ethertap(void)
9562306a36Sopenharmony_ci{
9662306a36Sopenharmony_ci	register_transport(&ethertap_transport);
9762306a36Sopenharmony_ci	return 0;
9862306a36Sopenharmony_ci}
9962306a36Sopenharmony_ci
10062306a36Sopenharmony_cilate_initcall(register_ethertap);
101