blob: 236a3dfc297d513dbbefbfab729932906455b0df [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * user-mode-linux networking multicast transport
3 * Copyright (C) 2001 by Harald Welte <laforge@gnumonks.org>
4 *
5 * based on the existing uml-networking code, which is
6 * Copyright (C) 2001 Lennert Buytenhek (buytenh@gnu.org) and
7 * James Leu (jleu@mindspring.net).
8 * Copyright (C) 2001 by various other people who didn't put their name here.
9 *
10 * Licensed under the GPL.
11 *
12 */
13
14#include <errno.h>
15#include <unistd.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070016#include <sys/socket.h>
17#include <sys/un.h>
18#include <sys/time.h>
19#include <netinet/in.h>
20#include "net_user.h"
21#include "mcast.h"
22#include "kern_util.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070023#include "user.h"
24#include "os.h"
Paolo 'Blaisorblade' Giarrussoc13e5692006-10-19 23:28:20 -070025#include "um_malloc.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070026
27#define MAX_PACKET (ETH_MAX_PACKET + ETH_HEADER_OTHER)
28
29static struct sockaddr_in *new_addr(char *addr, unsigned short port)
30{
31 struct sockaddr_in *sin;
32
Jeff Dikee4c4bf92007-07-15 23:38:56 -070033 sin = kmalloc(sizeof(struct sockaddr_in), UM_GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -070034 if(sin == NULL){
35 printk("new_addr: allocation of sockaddr_in failed\n");
Jeff Dike56bd1942007-05-06 14:51:02 -070036 return NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -070037 }
38 sin->sin_family = AF_INET;
39 sin->sin_addr.s_addr = in_aton(addr);
Jeff Dike7c00c312005-05-20 13:59:09 -070040 sin->sin_port = htons(port);
Jeff Dike56bd1942007-05-06 14:51:02 -070041 return sin;
Linus Torvalds1da177e2005-04-16 15:20:36 -070042}
43
Jeff Dikef34d9d22007-05-06 14:51:04 -070044static int mcast_user_init(void *data, void *dev)
Linus Torvalds1da177e2005-04-16 15:20:36 -070045{
46 struct mcast_data *pri = data;
47
48 pri->mcast_addr = new_addr(pri->addr, pri->port);
49 pri->dev = dev;
Jeff Dikef34d9d22007-05-06 14:51:04 -070050 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -070051}
52
Paolo 'Blaisorblade' Giarrusso83f4e8a2007-03-07 20:41:09 -080053static void mcast_remove(void *data)
54{
55 struct mcast_data *pri = data;
56
57 kfree(pri->mcast_addr);
58 pri->mcast_addr = NULL;
59}
60
Linus Torvalds1da177e2005-04-16 15:20:36 -070061static int mcast_open(void *data)
62{
63 struct mcast_data *pri = data;
64 struct sockaddr_in *sin = pri->mcast_addr;
65 struct ip_mreq mreq;
Paolo 'Blaisorblade' Giarrussoc50d2c42005-11-13 16:07:07 -080066 int fd, yes = 1, err = -EINVAL;
Linus Torvalds1da177e2005-04-16 15:20:36 -070067
68
Jeff Dike7c00c312005-05-20 13:59:09 -070069 if ((sin->sin_addr.s_addr == 0) || (sin->sin_port == 0))
Linus Torvalds1da177e2005-04-16 15:20:36 -070070 goto out;
Linus Torvalds1da177e2005-04-16 15:20:36 -070071
72 fd = socket(AF_INET, SOCK_DGRAM, 0);
Jeff Dike7c00c312005-05-20 13:59:09 -070073
Linus Torvalds1da177e2005-04-16 15:20:36 -070074 if (fd < 0){
Paolo 'Blaisorblade' Giarrussoc50d2c42005-11-13 16:07:07 -080075 err = -errno;
Linus Torvalds1da177e2005-04-16 15:20:36 -070076 printk("mcast_open : data socket failed, errno = %d\n",
77 errno);
Linus Torvalds1da177e2005-04-16 15:20:36 -070078 goto out;
79 }
80
81 if (setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &yes, sizeof(yes)) < 0) {
Paolo 'Blaisorblade' Giarrussoc50d2c42005-11-13 16:07:07 -080082 err = -errno;
Linus Torvalds1da177e2005-04-16 15:20:36 -070083 printk("mcast_open: SO_REUSEADDR failed, errno = %d\n",
84 errno);
Jeff Dike7c00c312005-05-20 13:59:09 -070085 goto out_close;
Linus Torvalds1da177e2005-04-16 15:20:36 -070086 }
87
88 /* set ttl according to config */
89 if (setsockopt(fd, SOL_IP, IP_MULTICAST_TTL, &pri->ttl,
90 sizeof(pri->ttl)) < 0) {
Paolo 'Blaisorblade' Giarrussoc50d2c42005-11-13 16:07:07 -080091 err = -errno;
Linus Torvalds1da177e2005-04-16 15:20:36 -070092 printk("mcast_open: IP_MULTICAST_TTL failed, error = %d\n",
93 errno);
Jeff Dike7c00c312005-05-20 13:59:09 -070094 goto out_close;
Linus Torvalds1da177e2005-04-16 15:20:36 -070095 }
96
97 /* set LOOP, so data does get fed back to local sockets */
98 if (setsockopt(fd, SOL_IP, IP_MULTICAST_LOOP, &yes, sizeof(yes)) < 0) {
Paolo 'Blaisorblade' Giarrussoc50d2c42005-11-13 16:07:07 -080099 err = -errno;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700100 printk("mcast_open: IP_MULTICAST_LOOP failed, error = %d\n",
101 errno);
Jeff Dike7c00c312005-05-20 13:59:09 -0700102 goto out_close;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700103 }
104
105 /* bind socket to mcast address */
106 if (bind(fd, (struct sockaddr *) sin, sizeof(*sin)) < 0) {
Jeff Dikeb4fd3102005-09-16 19:27:49 -0700107 err = -errno;
Paolo 'Blaisorblade' Giarrussoc50d2c42005-11-13 16:07:07 -0800108 printk("mcast_open : data bind failed, errno = %d\n", errno);
Jeff Dike7c00c312005-05-20 13:59:09 -0700109 goto out_close;
Jeff Dike56bd1942007-05-06 14:51:02 -0700110 }
111
Linus Torvalds1da177e2005-04-16 15:20:36 -0700112 /* subscribe to the multicast group */
113 mreq.imr_multiaddr.s_addr = sin->sin_addr.s_addr;
114 mreq.imr_interface.s_addr = 0;
115 if (setsockopt(fd, SOL_IP, IP_ADD_MEMBERSHIP,
116 &mreq, sizeof(mreq)) < 0) {
Paolo 'Blaisorblade' Giarrussoc50d2c42005-11-13 16:07:07 -0800117 err = -errno;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700118 printk("mcast_open: IP_ADD_MEMBERSHIP failed, error = %d\n",
119 errno);
120 printk("There appears not to be a multicast-capable network "
121 "interface on the host.\n");
122 printk("eth0 should be configured in order to use the "
123 "multicast transport.\n");
Paolo 'Blaisorblade' Giarrussoc50d2c42005-11-13 16:07:07 -0800124 goto out_close;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700125 }
126
Jeff Dike7c00c312005-05-20 13:59:09 -0700127 return fd;
128
129 out_close:
Paolo 'Blaisorblade' Giarrussoc50d2c42005-11-13 16:07:07 -0800130 os_close_file(fd);
Jeff Dikeb4fd3102005-09-16 19:27:49 -0700131 out:
Paolo 'Blaisorblade' Giarrussoc50d2c42005-11-13 16:07:07 -0800132 return err;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700133}
134
135static void mcast_close(int fd, void *data)
136{
137 struct ip_mreq mreq;
138 struct mcast_data *pri = data;
139 struct sockaddr_in *sin = pri->mcast_addr;
140
141 mreq.imr_multiaddr.s_addr = sin->sin_addr.s_addr;
142 mreq.imr_interface.s_addr = 0;
143 if (setsockopt(fd, SOL_IP, IP_DROP_MEMBERSHIP,
144 &mreq, sizeof(mreq)) < 0) {
145 printk("mcast_open: IP_DROP_MEMBERSHIP failed, error = %d\n",
146 errno);
147 }
148
149 os_close_file(fd);
150}
151
152int mcast_user_write(int fd, void *buf, int len, struct mcast_data *pri)
153{
154 struct sockaddr_in *data_addr = pri->mcast_addr;
155
Jeff Dike56bd1942007-05-06 14:51:02 -0700156 return net_sendto(fd, buf, len, data_addr, sizeof(*data_addr));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700157}
158
159static int mcast_set_mtu(int mtu, void *data)
160{
Jeff Dike56bd1942007-05-06 14:51:02 -0700161 return mtu;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700162}
163
Jeff Dike5e7672e2006-09-27 01:50:33 -0700164const struct net_user_info mcast_user_info = {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700165 .init = mcast_user_init,
166 .open = mcast_open,
167 .close = mcast_close,
Paolo 'Blaisorblade' Giarrusso83f4e8a2007-03-07 20:41:09 -0800168 .remove = mcast_remove,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700169 .set_mtu = mcast_set_mtu,
170 .add_address = NULL,
171 .delete_address = NULL,
172 .max_packet = MAX_PACKET - ETH_HEADER_OTHER
173};