2 * Copyright (C) 2001 Lennert Buytenhek (buytenh@gnu.org) and
3 * James Leu (jleu@mindspring.net).
4 * Copyright (C) 2001 by various other people who didn't put their name here.
5 * Licensed under the GPL.
11 #include <sys/socket.h>
16 #include "kern_util.h"
17 #include "user_util.h"
21 #define MAX_PACKET (ETH_MAX_PACKET + ETH_HEADER_OTHER)
23 enum request_type
{ REQ_NEW_CONTROL
};
25 #define SWITCH_MAGIC 0xfeedface
30 enum request_type type
;
31 struct sockaddr_un sock
;
34 static struct sockaddr_un
*new_addr(void *name
, int len
)
36 struct sockaddr_un
*sun
;
38 sun
= um_kmalloc(sizeof(struct sockaddr_un
));
40 printk("new_addr: allocation of sockaddr_un failed\n");
43 sun
->sun_family
= AF_UNIX
;
44 memcpy(sun
->sun_path
, name
, len
);
48 static int connect_to_switch(struct daemon_data
*pri
)
50 struct sockaddr_un
*ctl_addr
= pri
->ctl_addr
;
51 struct sockaddr_un
*local_addr
= pri
->local_addr
;
52 struct sockaddr_un
*sun
;
53 struct request_v3 req
;
56 pri
->control
= socket(AF_UNIX
, SOCK_STREAM
, 0);
58 printk("daemon_open : control socket failed, errno = %d\n",
63 if(connect(pri
->control
, (struct sockaddr
*) ctl_addr
,
64 sizeof(*ctl_addr
)) < 0){
65 printk("daemon_open : control connect failed, errno = %d\n",
71 fd
= socket(AF_UNIX
, SOCK_DGRAM
, 0);
73 printk("daemon_open : data socket failed, errno = %d\n",
78 if(bind(fd
, (struct sockaddr
*) local_addr
, sizeof(*local_addr
)) < 0){
79 printk("daemon_open : data bind failed, errno = %d\n",
85 sun
= um_kmalloc(sizeof(struct sockaddr_un
));
87 printk("new_addr: allocation of sockaddr_un failed\n");
92 req
.magic
= SWITCH_MAGIC
;
93 req
.version
= SWITCH_VERSION
;
94 req
.type
= REQ_NEW_CONTROL
;
95 req
.sock
= *local_addr
;
96 n
= os_write_file(pri
->control
, &req
, sizeof(req
));
98 printk("daemon_open : control setup request failed, err = %d\n",
104 n
= os_read_file(pri
->control
, sun
, sizeof(*sun
));
105 if(n
!= sizeof(*sun
)){
106 printk("daemon_open : read of data socket failed, err = %d\n",
112 pri
->data_addr
= sun
;
118 os_close_file(pri
->control
);
122 static void daemon_user_init(void *data
, void *dev
)
124 struct daemon_data
*pri
= data
;
132 if(!strcmp(pri
->sock_type
, "unix"))
133 pri
->ctl_addr
= new_addr(pri
->ctl_sock
,
134 strlen(pri
->ctl_sock
) + 1);
136 name
.pid
= os_getpid();
137 gettimeofday(&tv
, NULL
);
138 name
.usecs
= tv
.tv_usec
;
139 pri
->local_addr
= new_addr(&name
, sizeof(name
));
141 pri
->fd
= connect_to_switch(pri
);
143 kfree(pri
->local_addr
);
144 pri
->local_addr
= NULL
;
148 static int daemon_open(void *data
)
150 struct daemon_data
*pri
= data
;
154 static void daemon_remove(void *data
)
156 struct daemon_data
*pri
= data
;
158 os_close_file(pri
->fd
);
159 os_close_file(pri
->control
);
160 if(pri
->data_addr
!= NULL
) kfree(pri
->data_addr
);
161 if(pri
->ctl_addr
!= NULL
) kfree(pri
->ctl_addr
);
162 if(pri
->local_addr
!= NULL
) kfree(pri
->local_addr
);
165 int daemon_user_write(int fd
, void *buf
, int len
, struct daemon_data
*pri
)
167 struct sockaddr_un
*data_addr
= pri
->data_addr
;
169 return(net_sendto(fd
, buf
, len
, data_addr
, sizeof(*data_addr
)));
172 static int daemon_set_mtu(int mtu
, void *data
)
177 struct net_user_info daemon_user_info
= {
178 .init
= daemon_user_init
,
181 .remove
= daemon_remove
,
182 .set_mtu
= daemon_set_mtu
,
184 .delete_address
= NULL
,
185 .max_packet
= MAX_PACKET
- ETH_HEADER_OTHER
189 * Overrides for Emacs so that we follow Linus's tabbing style.
190 * Emacs will notice this stuff at the end of the file and automatically
191 * adjust the settings for this buffer only. This must remain at the end
193 * ---------------------------------------------------------------------------
195 * c-file-style: "linux"