2 * cbcp - Call Back Configuration Protocol.
4 * Copyright (c) 1995 Pedro Roque Marques
7 * Redistribution and use in source and binary forms are permitted
8 * provided that the above copyright notice and this paragraph are
9 * duplicated in all such forms and that any documentation,
10 * advertising materials, and other materials related to such
11 * distribution and use acknowledge that the software was developed
12 * by Pedro Roque Marques. The name of the author may not be used to
13 * endorse or promote products derived from this software without
14 * specific prior written permission.
16 * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
17 * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
18 * WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR A PARTICULAR PURPOSE.
22 static char rcsid
[] = "$FreeBSD$";
27 #include <sys/types.h>
38 * Protocol entry points.
40 static void cbcp_init(int unit
);
41 static void cbcp_open(int unit
);
42 static void cbcp_lowerup(int unit
);
43 static void cbcp_input(int unit
, u_char
*pkt
, int len
);
44 static void cbcp_protrej(int unit
);
45 static int cbcp_printpkt(u_char
*pkt
, int len
,
46 void (*printer
)(void *, char *, ...),
49 struct protent cbcp_protent
= {
67 cbcp_state cbcp
[NUM_PPP
];
69 /* internal prototypes */
71 static void cbcp_recvreq(cbcp_state
*us
, char *pckt
, int len
);
72 static void cbcp_resp(cbcp_state
*us
);
73 static void cbcp_up(cbcp_state
*us
);
74 static void cbcp_recvack(cbcp_state
*us
, char *pckt
, int len
);
75 static void cbcp_send(cbcp_state
*us
, u_char code
, u_char
*buf
, int len
);
85 memset(us
, 0, sizeof(cbcp_state
));
87 us
->us_type
|= (1 << CB_CONF_NO
);
90 /* lower layer is up */
95 cbcp_state
*us
= &cbcp
[iface
];
97 syslog(LOG_DEBUG
, "cbcp_lowerup");
98 syslog(LOG_DEBUG
, "want: %d", us
->us_type
);
100 if (us
->us_type
== CB_CONF_USER
)
101 syslog(LOG_DEBUG
, "phone no: %s", us
->us_number
);
108 syslog(LOG_DEBUG
, "cbcp_open");
111 /* process an incomming packet */
113 cbcp_input(unit
, inpacket
, pktlen
)
122 cbcp_state
*us
= &cbcp
[unit
];
126 if (pktlen
< CBCP_MINLEN
) {
127 syslog(LOG_ERR
, "CBCP packet is too small");
135 if (len
< CBCP_MINLEN
|| len
> pktlen
) {
136 syslog(LOG_ERR
, "CBCP packet: invalid length");
145 cbcp_recvreq(us
, inp
, len
);
149 syslog(LOG_DEBUG
, "CBCP_RESP received");
154 syslog(LOG_DEBUG
, "id doesn't match: expected %d recv %d",
157 cbcp_recvack(us
, inp
, len
);
165 /* protocol was rejected by foe */
166 void cbcp_protrej(int iface
)
170 char *cbcp_codenames
[] = {
171 "Request", "Response", "Ack"
174 char *cbcp_optionnames
[] = {
181 /* pretty print a packet */
183 cbcp_printpkt(p
, plen
, printer
, arg
)
186 void (*printer
)(void *, char *, ...);
189 int code
, opt
, id
, len
, olen
, delay
;
192 if (plen
< HEADERLEN
)
198 if (len
< HEADERLEN
|| len
> plen
)
201 if (code
>= 1 && code
<= sizeof(cbcp_codenames
) / sizeof(char *))
202 printer(arg
, " %s", cbcp_codenames
[code
-1]);
204 printer(arg
, " code=0x%x", code
);
206 printer(arg
, " id=0x%x", id
);
217 if (olen
< 2 || olen
> len
) {
224 if (opt
>= 1 && opt
<= sizeof(cbcp_optionnames
) / sizeof(char *))
225 printer(arg
, " %s", cbcp_optionnames
[opt
-1]);
227 printer(arg
, " option=0x%x", opt
);
231 printer(arg
, " delay = %d", delay
);
239 memcpy(str
, p
, olen
- 4);
241 printer(arg
, " number = %s", str
);
251 for (; len
> 0; --len
) {
253 printer(arg
, " %.2x", code
);
259 /* received CBCP request */
261 cbcp_recvreq(us
, pckt
, pcktlen
)
266 u_char type
, opt_len
, delay
, addr_type
;
273 syslog(LOG_DEBUG
, "length: %d", len
);
276 GETCHAR(opt_len
, pckt
);
283 GETCHAR(delay
, pckt
);
285 us
->us_allowed
|= (1 << type
);
289 syslog(LOG_DEBUG
, "no callback allowed");
293 syslog(LOG_DEBUG
, "user callback allowed");
295 GETCHAR(addr_type
, pckt
);
296 memcpy(address
, pckt
, opt_len
- 4);
297 address
[opt_len
- 4] = 0;
299 syslog(LOG_DEBUG
, "address: %s", address
);
304 syslog(LOG_DEBUG
, "user admin defined allowed");
324 cb_type
= us
->us_allowed
& us
->us_type
;
325 syslog(LOG_DEBUG
, "cbcp_resp cb_type=%d", cb_type
);
329 lcp_down(us
->us_unit
);
332 if (cb_type
& ( 1 << CB_CONF_USER
) ) {
333 syslog(LOG_DEBUG
, "cbcp_resp CONF_USER");
334 PUTCHAR(CB_CONF_USER
, bufp
);
335 len
= 3 + 1 + strlen(us
->us_number
) + 1;
337 PUTCHAR(5, bufp
); /* delay */
339 BCOPY(us
->us_number
, bufp
, strlen(us
->us_number
) + 1);
340 cbcp_send(us
, CBCP_RESP
, buf
, len
);
344 if (cb_type
& ( 1 << CB_CONF_ADMIN
) ) {
345 syslog(LOG_DEBUG
, "cbcp_resp CONF_ADMIN");
346 PUTCHAR(CB_CONF_ADMIN
, bufp
);
349 PUTCHAR(5, bufp
); /* delay */
350 cbcp_send(us
, CBCP_RESP
, buf
, len
);
354 if (cb_type
& ( 1 << CB_CONF_NO
) ) {
355 syslog(LOG_DEBUG
, "cbcp_resp CONF_NO");
356 PUTCHAR(CB_CONF_NO
, bufp
);
359 cbcp_send(us
, CBCP_RESP
, buf
, len
);
360 (*ipcp_protent
.open
)(us
->us_unit
);
366 cbcp_send(us
, code
, buf
, len
)
375 outp
= outpacket_buf
;
379 MAKEHEADER(outp
, PPP_CBCP
);
382 PUTCHAR(us
->us_id
, outp
);
383 PUTSHORT(outlen
, outp
);
386 BCOPY(buf
, outp
, len
);
388 output(us
->us_unit
, outpacket_buf
, outlen
+ PPP_HDRLEN
);
392 cbcp_recvack(us
, pckt
, len
)
397 u_char type
, delay
, addr_type
;
403 GETCHAR(opt_len
, pckt
);
409 GETCHAR(delay
, pckt
);
412 GETCHAR(addr_type
, pckt
);
413 memcpy(address
, pckt
, opt_len
- 4);
414 address
[opt_len
- 4] = 0;
416 syslog(LOG_DEBUG
, "peer will call: %s", address
);
425 /* ok peer will do callback */
431 lcp_close(0, "Call me back, please");