1 /* pppoatm.c - pppd plugin to implement PPPoATM protocol.
3 * Copyright 2000 Mitchell Blank Jr.
4 * Based in part on work from Jens Axboe and Paul Mackerras.
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version
9 * 2 of the License, or (at your option) any later version.
12 #include "pathnames.h"
13 #include "fsm.h" /* Needed for lcp.h to include cleanly */
16 #include <linux/atmdev.h>
17 #include <linux/atmppp.h>
20 #include <sys/ioctl.h>
22 static struct sockaddr_atmpvc pvcaddr;
23 static char *qosstr = NULL;
24 static int pppoatm_accept = 0;
27 static int device_got_set = 0;
28 static int pppoatm_max_mtu, pppoatm_max_mru;
31 struct channel pppoatm_channel;
32 static int fd_pppoa=0;
34 int setdevname_pppoatm(const char *cp);
35 //static int setdevname_pppoatm(const char *cp);
36 static void set_line_discipline_pppoatm(int fd);
39 static option_t my_options[] = {
41 { "accept", o_bool, &pppoatm_accept,
42 "set PPPoATM socket to accept incoming connections", 1 },
44 { "llc-encaps", o_bool, &llc_encaps,
45 "use LLC encapsulation for PPPoATM", 1},
46 { "vc-encaps", o_bool, &vc_encaps,
47 "use VC multiplexing for PPPoATM (default)", 1},
48 { "qos", o_string, &qosstr,
49 "set QoS for PPPoATM connection", 1},
54 * -1 if there's a problem with setting the device
55 * 0 if we can't parse "cp" as a valid name of a device
56 * 1 if "cp" is a reasonable thing to name a device
57 * Note that we don't actually open the device at this point
58 * We do need to fill in:
59 * devnam: a string representation of the device
60 * devstat: a stat structure of the device. In this case
61 * we're not opening a device, so we just make sure
62 * to set up S_ISCHR(devstat.st_mode) != 1, so we
63 * don't get confused that we're on stdin.
66 int setdevname_pppoatm(const char *cp)
67 //static int setdevname_pppoatm(const char *cp)
70 struct sockaddr_atmpvc addr;
72 // extern struct stat devstat;
76 info("PPPoATM setdevname_pppoatm");
77 memset(&addr, 0, sizeof addr);
78 if (text2atm(cp, (struct sockaddr *) &addr, sizeof(addr),
79 T2A_PVC | T2A_NAME) < 0)
85 memcpy(&pvcaddr, &addr, sizeof pvcaddr);
86 strlcpy(devnam, cp, sizeof devnam);
88 // devstat.st_mode = S_IFSOCK;
91 the_channel = &pppoatm_channel;
93 lcp_wantoptions[0].neg_accompression = 0;
94 lcp_allowoptions[0].neg_accompression = 0;
95 lcp_wantoptions[0].neg_asyncmap = 0;
96 lcp_allowoptions[0].neg_asyncmap = 0;
97 lcp_wantoptions[0].neg_pcompression = 0;
100 info("PPPoATM setdevname_pppoatm - SUCCESS");
105 static int setspeed_pppoatm(const char *cp)
110 static int options_for_pppoatm(void)
113 // return options_from_devfile(_PATH_ATMOPT, devnam);
117 #define pppoatm_overhead() (llc_encaps ? 6 : 2)
120 static disconnect_pppoatm(int fd)
122 unsigned char ppp_disconn[64];
123 unsigned char * outp;
126 MAKEHEADER(outp, PPP_LCP); // 2 bytes
127 PUTCHAR(5, outp); // TERMREQ==5 // 1 byte
128 PUTCHAR(2, outp); // id=02 // 1 byte
129 PUTSHORT(4, outp); // HEADERLEN==4 in fsm.h // 2 byte
131 write(fd, ppp_disconn+2, 6);
135 static void no_device_given_pppoatm(void)
137 fatal("No vpi.vci specified");
140 static int open_device_pppoatm(void)
147 no_device_given_pppoatm();
148 fd = socket(AF_ATMPVC, SOCK_DGRAM, 0);
150 fatal("failed to create socket: %m");
153 memset(&qos, 0, sizeof qos);
154 qos.txtp.traffic_class = qos.rxtp.traffic_class = ATM_UBR;
155 /* TODO: support simplified QoS setting */
157 if (text2qos(qosstr, &qos, 0))
158 fatal("Can't parse QoS: \"%s\"");
159 qos.txtp.max_sdu = lcp_allowoptions[0].mru + pppoatm_overhead();
160 qos.rxtp.max_sdu = lcp_wantoptions[0].mru + pppoatm_overhead();
162 if (setsockopt(fd, SOL_ATM, SO_ATMQOS, &qos, sizeof(qos)) < 0)
163 fatal("setsockopt(SO_ATMQOS): %m");
164 /* TODO: accept on SVCs... */
165 if (connect(fd, (struct sockaddr *) &pvcaddr,
166 sizeof(struct sockaddr_atmpvc)))
167 fatal("connect(%s): %m", devnam);
168 pppoatm_max_mtu = lcp_allowoptions[0].mru;
169 pppoatm_max_mru = lcp_wantoptions[0].mru;
171 set_line_discipline_pppoatm(fd);
177 disconnect_pppoatm(fd);
185 static void post_open_setup_pppoatm(int fd)
190 static void pre_close_restore_pppoatm(int fd)
196 static void close_device_pppoatm(void)
204 static void set_line_discipline_pppoatm(int fd)
206 struct atm_backend_ppp be;
207 be.backend_num = ATM_BACKEND_PPP;
209 be.encaps = PPPOATM_ENCAPS_VC;
211 be.encaps = PPPOATM_ENCAPS_LLC;
213 be.encaps = PPPOATM_ENCAPS_AUTODETECT;
214 if (ioctl(fd, ATM_SETBACKEND, &be) < 0)
215 fatal("ioctl(ATM_SETBACKEND): %m");
218 static void reset_line_discipline_pppoatm(int fd)
220 atm_backend_t be = ATM_BACKEND_RAW;
221 /* 2.4 doesn't support this yet */
222 (void) ioctl(fd, ATM_SETBACKEND, &be);
226 static void send_config_pppoatm(int mtu, u_int32_t asyncmap,
227 int pcomp, int accomp)
228 //static void send_config_pppoatm(int unit, int mtu, u_int32_t asyncmap,
229 // int pcomp, int accomp)
234 if (mtu > pppoatm_max_mtu)
235 error("Couldn't increase MTU to %d", mtu);
236 sock = socket(AF_INET, SOCK_DGRAM, 0);
238 fatal("Couldn't create IP socket: %m");
239 strlcpy(ifr.ifr_name, ifname, sizeof(ifr.ifr_name));
241 if (ioctl(sock, SIOCSIFMTU, (caddr_t) &ifr) < 0)
242 fatal("ioctl(SIOCSIFMTU): %m");
247 static void recv_config_pppoatm(int mru, u_int32_t asyncmap,
248 int pcomp, int accomp)
249 //static void recv_config_pppoatm(int unit, int mru, u_int32_t asyncmap,
250 // int pcomp, int accomp)
253 if (mru > pppoatm_max_mru)
254 error("Couldn't increase MRU to %d", mru);
257 static void set_xaccm_pppoatm(int unit, ext_accm accm)
264 void plugin_init(void)
267 extern int new_style_driver; /* From sys-linux.c */
269 static char *bad_options[] = {
271 "default-asyncmap", "-am", "asyncmap", "-as", "escape",
273 "crtscts", "-crtscts", "nocrtscts",
274 "cdtrcts", "nocdtrcts",
276 "modem", "local", "sync",
279 if (!new_style_driver)
280 fatal("Kernel doesn't support ppp_generic - "
281 "needed for PPPoATM");
283 fatal("No PPPoATM support on this OS");
285 info("PPPoATM plugin_init");
286 add_options(my_options);
287 add_devname_class(setdevname_pppoatm);
288 setspeed_hook = setspeed_pppoatm;
289 options_for_device_hook = options_for_pppoatm;
290 open_device_hook = open_device_pppoatm;
291 post_open_setup_hook = post_open_setup_pppoatm;
292 pre_close_restore_hook = pre_close_restore_pppoatm;
293 no_device_given_hook = no_device_given_pppoatm;
294 set_line_discipline_hook = set_line_discipline_pppoatm;
295 reset_line_discipline_hook = reset_line_discipline_pppoatm;
296 send_config_hook = send_config_pppoatm;
297 recv_config_hook = recv_config_pppoatm;
298 set_xaccm_hook = set_xaccm_pppoatm;
301 for (a = bad_options; *a != NULL; a++)
305 lcp_wantoptions[0].neg_accompression = 0;
306 lcp_allowoptions[0].neg_accompression = 0;
307 lcp_wantoptions[0].neg_asyncmap = 0;
308 lcp_allowoptions[0].neg_asyncmap = 0;
309 lcp_wantoptions[0].neg_pcompression = 0;
315 struct channel pppoatm_channel = {
317 process_extra_options: NULL,
319 connect: &open_device_pppoatm,
320 disconnect: &close_device_pppoatm,
321 establish_ppp: &generic_establish_ppp,
322 disestablish_ppp: &generic_disestablish_ppp,
323 send_config: &send_config_pppoatm,
324 recv_config: &recv_config_pppoatm,