* Copyright (c) 1983, 1988 Regents of the University of California.
* %sccs.include.redist.c%
static char sccsid
[] = "@(#)route.c 5.23 (Berkeley) %G%";
#include <net/if_types.h>
(kread((off_t)(p), (char *)&(d), sizeof (d)))
* Definitions for showing gateway flags.
static struct sockaddr
*kgetsa
__P((struct sockaddr
*));
static void p_tree
__P((struct radix_node
*));
static void p_rtnode
__P(());
static void ntreestuff
__P(());
static void np_rtentry
__P((struct rt_msghdr
*));
static void p_sockaddr
__P((struct sockaddr
*, int, int));
static void p_flags
__P((int, char *));
static void p_rtentry
__P((struct rtentry
*));
struct radix_node_head
*rnh
, head
;
printf("Routing tables\n");
if (Aflag
== 0 && NewTree
)
printf("rt_tables: symbol not in namelist\n");
for (i
= 0; i
<= AF_MAX
; i
++) {
if ((rnh
= rt_tables
[i
]) == 0)
p_tree(head
.rnh_treetop
);
} else if (af
== AF_UNSPEC
|| af
== i
) {
p_tree(head
.rnh_treetop
);
* Print address family header before a section of the routing table.
printf("\n%s:\n", afname
);
printf("\nProtocol Family %d:\n", af
);
/* column widths; each followed by one space */
#define WID_DST 16 /* width of destination column */
#define WID_GW 18 /* width of gateway column */
* Print header for routing table columns.
printf("%-8.8s ","Address");
printf("%-*.*s %-*.*s %-6.6s %6.6s%8.8s %s\n",
WID_DST
, WID_DST
, "Destination",
WID_GW
, WID_GW
, "Gateway",
"Flags", "Refs", "Use", "Interface");
register struct sockaddr
*dst
;
if (pt_u
.u_sa
.sa_len
> sizeof (pt_u
.u_sa
))
kread((off_t
)dst
, (char *)pt_u
.u_data
, pt_u
.u_sa
.sa_len
);
if (rnode
.rn_flags
& RNF_ROOT
) {
rnode
.rn_dupedkey
? " =>\n" : "\n");
p_sockaddr(kgetsa((struct sockaddr
*)rnode
.rn_key
),
if (rn
= rnode
.rn_dupedkey
)
struct radix_mask
*rm
= rnode
.rn_mklist
;
p_sockaddr(kgetsa((struct sockaddr
*)rnode
.rn_mask
),
sprintf(nbuf
, "(%d)", rnode
.rn_b
);
printf("%6.6s %8.8x : %8.8x", nbuf
, rnode
.rn_l
, rnode
.rn_r
);
sprintf(nbuf
, " %d refs, ", rmask
.rm_refs
);
printf(" mk = %8.8x {(%d),%s",
rm
, -1 - rmask
.rm_b
, rmask
.rm_refs
? nbuf
: " ");
p_sockaddr(kgetsa((struct sockaddr
*)rmask
.rm_mask
), 0, -1);
if (rm
= rmask
.rm_mklist
)
register struct rt_msghdr
*rtm
;
if ((needed
= getkerninfo(KINFO_RT_DUMP
, 0, 0, 0)) < 0)
{ perror("route-getkerninfo-estimate"); exit(1);}
if ((buf
= malloc(needed
)) == 0)
{ printf("out of space\n"); exit(1);}
if (getkerninfo(KINFO_RT_DUMP
, buf
, &needed
, 0) < 0)
{ perror("getkerninfo of routing table"); exit(1);}
for (next
= buf
; next
< lim
; next
+= rtm
->rtm_msglen
) {
rtm
= (struct rt_msghdr
*)next
;
register struct rt_msghdr
*rtm
;
register struct sockaddr
*sa
= (struct sockaddr
*)(rtm
+ 1);
static int masks_done
, banner_printed
;
int af
= 0, interesting
= RTF_UP
| RTF_GATEWAY
| RTF_HOST
;
/* for the moment, netmasks are skipped over */
if (rtm
->rtm_addrs
!= RTA_DST
) {
if (rtm
->rtm_addrs
== RTA_DST
)
p_sockaddr(sa
, rtm
->rtm_flags
, 16);
sa
->sa_len
= sizeof(long);
sa
= (struct sockaddr
*)(sa
->sa_len
+ (char *)sa
);
p_flags(rtm
->rtm_flags
& interesting
, "%-6.6s ");
p_sockaddr(sa
, flags
, width
)
char workbuf
[128], *cplim
;
register char *cp
= workbuf
;
register struct sockaddr_in
*sin
= (struct sockaddr_in
*)sa
;
cp
= (sin
->sin_addr
.s_addr
== 0) ? "default" :
routename(sin
->sin_addr
.s_addr
) :
netname(sin
->sin_addr
.s_addr
, 0L));
register struct sockaddr_dl
*sdl
= (struct sockaddr_dl
*)sa
;
if (sdl
->sdl_nlen
== 0 && sdl
->sdl_alen
== 0 &&
(void) sprintf(workbuf
, "link#%d", sdl
->sdl_index
);
else switch (sdl
->sdl_type
) {
register u_char
*lla
= (u_char
*)sdl
->sdl_data
+
for (i
= 0; i
< sdl
->sdl_alen
; i
++, lla
++) {
cp
+= sprintf(cp
, "%s%x", cplim
, *lla
);
register u_short
*s
= ((u_short
*)sa
->sa_data
), *slim
;
slim
= (u_short
*) sa
+ ((sa
->sa_len
+ sizeof(u_short
) - 1) /
cplim
= cp
+ sizeof(workbuf
) - 6;
cp
+= sprintf(cp
, "(%d)", sa
->sa_family
);
while (s
< slim
&& cp
< cplim
)
cp
+= sprintf(cp
, " %x", *s
++);
printf("%-*s ", width
, cp
);
printf("%-*.*s ", width
, width
, cp
);
register struct bits
*p
= bits
;
for (flags
= name
; p
->b_mask
; p
++)
register struct rtentry
*rt
;
static struct ifnet ifnet
, *lastif
;
p_sockaddr(kgetsa(rt_key(rt
)), rt
->rt_flags
, WID_DST
);
p_sockaddr(kgetsa(rt
->rt_gateway
), 0, WID_GW
);
p_flags(rt
->rt_flags
, "%-6.6s ");
printf("%6d %8d ", rt
->rt_refcnt
, rt
->rt_use
);
if (rt
->rt_ifp
!= lastif
) {
kread((off_t
)ifnet
.if_name
, name
, 16);
printf(" %.15s%d%s", name
, ifnet
.if_unit
,
rt
->rt_nodes
[0].rn_dupedkey
? " =>" : "");
static char line
[MAXHOSTNAMELEN
+ 1];
static char domain
[MAXHOSTNAMELEN
+ 1];
if (gethostname(domain
, MAXHOSTNAMELEN
) == 0 &&
(cp
= index(domain
, '.')))
(void) strcpy(domain
, cp
+ 1);
hp
= gethostbyaddr((char *)&in
, sizeof (struct in_addr
),
if ((cp
= index(hp
->h_name
, '.')) &&
strncpy(line
, cp
, sizeof(line
) - 1);
#define C(x) ((x) & 0xff)
sprintf(line
, "%u.%u.%u.%u",
C(in
>> 24), C(in
>> 16), C(in
>> 8), C(in
));
* Return the name of the network whose address is given.
* The address is assumed to be that of a net or subnet, not a host.
static char line
[MAXHOSTNAMELEN
+ 1];
} else if (IN_CLASSB(i
)) {
* If there are more bits than the standard mask
* would suggest, subnets must be in use.
* Guess at the subnet mask, assuming reasonable
mask
= (long)mask
>> subnetshift
;
np
= getnetbyaddr(net
, AF_INET
);
strncpy(line
, cp
, sizeof(line
) - 1);
else if ((i
& 0xffffff) == 0)
sprintf(line
, "%u", C(i
>> 24));
else if ((i
& 0xffff) == 0)
sprintf(line
, "%u.%u", C(i
>> 24) , C(i
>> 16));
else if ((i
& 0xff) == 0)
sprintf(line
, "%u.%u.%u", C(i
>> 24), C(i
>> 16), C(i
>> 8));
sprintf(line
, "%u.%u.%u.%u", C(i
>> 24),
C(i
>> 16), C(i
>> 8), C(i
));
* Print routing statistics
printf("rtstat: symbol not in namelist\n");
kread(off
, (char *)&rtstat
, sizeof (rtstat
));
printf("\t%u bad routing redirect%s\n",
rtstat
.rts_badredirect
, plural(rtstat
.rts_badredirect
));
printf("\t%u dynamically created route%s\n",
rtstat
.rts_dynamic
, plural(rtstat
.rts_dynamic
));
printf("\t%u new gateway%s due to redirects\n",
rtstat
.rts_newgateway
, plural(rtstat
.rts_newgateway
));
printf("\t%u destination%s found unreachable\n",
rtstat
.rts_unreach
, plural(rtstat
.rts_unreach
));
printf("\t%u use%s of a wildcard route\n",
rtstat
.rts_wildcard
, plural(rtstat
.rts_wildcard
));
short ns_nullh
[] = {0,0,0};
short ns_bh
[] = {-1,-1,-1};
register struct sockaddr
*sa
;
register struct sockaddr_ns
*sns
= (struct sockaddr_ns
*)sa
;
union { union ns_net net_e
; u_long long_e
; } net
;
static char mybuf
[50], cport
[10], chost
[25];
register char *p
; register u_char
*q
;
port
= ntohs(work
.x_port
);
if (ns_nullhost(work
) && net
.long_e
== 0) {
sprintf(mybuf
, "*.%xH", port
);
if (bcmp(ns_bh
, work
.x_host
.c_host
, 6) == 0) {
} else if (bcmp(ns_nullh
, work
.x_host
.c_host
, 6) == 0) {
sprintf(chost
, "%02x%02x%02x%02x%02x%02xH",
q
[0], q
[1], q
[2], q
[3], q
[4], q
[5]);
for (p
= chost
; *p
== '0' && p
< chost
+ 12; p
++)
sprintf(cport
, ".%xH", htons(port
));
sprintf(mybuf
,"%xH.%s%s", ntohl(net
.long_e
), host
, cport
);
register struct sockaddr_ns
*sns
= (struct sockaddr_ns
*)sa
;
static union ns_net ns_zeronet
;
work
.sns_addr
.x_port
= 0;
work
.sns_addr
.x_net
= ns_zeronet
;
p
= ns_print((struct sockaddr
*)&work
);
if (strncmp("0H.", p
, 3) == 0) p
+= 3;
for (; *p
; p
++) switch (*p
) {
case 'a': case 'b': case 'c': case 'd': case 'e': case 'f':