2 * Copyright (c) 2004-2005 Gleb Smirnoff <glebius@FreeBSD.org>
3 * Copyright (c) 2001-2003 Roman V. Palagin <romanp@unshadow.net>
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions
9 * 1. Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * 2. Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
15 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
16 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
17 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
18 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
19 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
20 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
21 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
22 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
23 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
24 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27 * $SourceForge: flowctl.c,v 1.15 2004/08/31 20:24:58 glebius Exp $
31 static const char rcs_id
[] =
35 #include <sys/types.h>
37 #include <sys/socket.h>
38 #include <sys/queue.h>
41 #include <netinet/in.h>
43 #include <arpa/inet.h>
52 #include <netgraph/netflow/ng_netflow.h>
54 #define CISCO_SH_FLOW_HEADER "SrcIf SrcIPaddress DstIf DstIPaddress Pr SrcP DstP Pkts\n"
55 #define CISCO_SH_FLOW "%-13s %-15s %-13s %-15s %2u %4.4x %4.4x %6lu\n"
57 #define CISCO_SH_VERB_FLOW_HEADER "SrcIf SrcIPaddress DstIf DstIPaddress Pr TOS Flgs Pkts\n" \
58 "Port Msk AS Port Msk AS NextHop B/Pk Active\n"
60 #define CISCO_SH_VERB_FLOW "%-14s %-15s %-14s %-15s %2u %3x %4x %6lu\n" \
61 "%4.4x /%-2u %-5u %4.4x /%-2u %-5u %-15s %9u %8u\n\n"
63 static int flow_cache_print(struct ngnf_flows
*recs
);
64 static int flow_cache_print_verbose(struct ngnf_flows
*recs
);
65 static int ctl_show(int, char **);
66 static void help(void);
67 static void execute_command(int, char **);
71 int (*cmd_func
)(int argc
, char **argv
);
74 struct ip_ctl_cmd cmds
[] = {
80 char ng_nodename
[NG_PATHLEN
+ 1];
83 main(int argc
, char **argv
)
86 char sname
[NG_NODESIZ
];
87 int rcvbuf
= SORCVBUF_SIZE
;
91 while ((c
= getopt(argc
, argv
, "d:")) != -1) {
93 case 'd': /* set libnetgraph debug level. */
94 NgSetDebug(atoi(optarg
));
107 snprintf(ng_nodename
, sizeof(ng_nodename
), "%s:", ng_name
);
109 /* create control socket. */
110 snprintf(sname
, sizeof(sname
), "flowctl%i", getpid());
112 if (NgMkSockNode(sname
, &cs
, NULL
) == -1)
113 err(1, "NgMkSockNode");
115 /* set receive buffer size */
116 if (setsockopt(cs
, SOL_SOCKET
, SO_RCVBUF
, &rcvbuf
, sizeof(int)) == -1)
117 err(1, "setsockopt(SOL_SOCKET, SO_RCVBUF)");
119 /* parse and execute command */
120 execute_command(argc
, argv
);
128 execute_command(int argc
, char **argv
)
135 for (i
= 0; cmds
[i
].cmd_name
!= NULL
; i
++)
136 if (!strncmp(argv
[0], cmds
[i
].cmd_name
, strlen(argv
[0]))) {
138 errx(1, "ambiguous command: %s", argv
[0]);
142 errx(1, "bad command: %s", argv
[0]);
145 (*cmds
[cindex
].cmd_func
)(argc
, argv
);
149 ctl_show(int argc
, char **argv
)
151 struct ng_mesg
*ng_mesg
;
152 struct ngnf_flows
*data
;
153 char path
[NG_PATHLEN
+ 1];
154 int token
, nread
, last
= 0;
157 if (argc
> 0 && !strncmp(argv
[0], "verbose", strlen(argv
[0])))
160 ng_mesg
= alloca(SORCVBUF_SIZE
);
163 printf(CISCO_SH_VERB_FLOW_HEADER
);
165 printf(CISCO_SH_FLOW_HEADER
);
168 /* request set of accounting records */
169 token
= NgSendMsg(cs
, ng_nodename
, NGM_NETFLOW_COOKIE
,
170 NGM_NETFLOW_SHOW
, (void *)&last
, sizeof(last
));
172 err(1, "NgSendMsg(NGM_NETFLOW_SHOW)");
175 nread
= NgRecvMsg(cs
, ng_mesg
, SORCVBUF_SIZE
, path
);
177 err(1, "NgRecvMsg() failed");
179 if (ng_mesg
->header
.token
!= token
)
180 err(1, "NgRecvMsg(NGM_NETFLOW_SHOW): token mismatch");
182 data
= (struct ngnf_flows
*)ng_mesg
->data
;
183 if ((ng_mesg
->header
.arglen
< (sizeof(*data
))) ||
184 (ng_mesg
->header
.arglen
< (sizeof(*data
) +
185 (data
->nentries
* sizeof(struct flow_entry_data
)))))
186 err(1, "NgRecvMsg(NGM_NETFLOW_SHOW): arglen too small");
189 (void )flow_cache_print_verbose(data
);
191 (void )flow_cache_print(data
);
203 flow_cache_print(struct ngnf_flows
*recs
)
205 struct flow_entry_data
*fle
;
206 char src
[INET_ADDRSTRLEN
], dst
[INET_ADDRSTRLEN
];
207 char src_if
[IFNAMSIZ
], dst_if
[IFNAMSIZ
];
211 if (recs
->nentries
== 0)
215 for (i
= 0; i
< recs
->nentries
; i
++, fle
++) {
216 inet_ntop(AF_INET
, &fle
->r
.r_src
, src
, sizeof(src
));
217 inet_ntop(AF_INET
, &fle
->r
.r_dst
, dst
, sizeof(dst
));
218 printf(CISCO_SH_FLOW
,
219 if_indextoname(fle
->fle_i_ifx
, src_if
),
221 if_indextoname(fle
->fle_o_ifx
, dst_if
),
224 ntohs(fle
->r
.r_sport
),
225 ntohs(fle
->r
.r_dport
),
234 flow_cache_print_verbose(struct ngnf_flows
*recs
)
236 struct flow_entry_data
*fle
;
237 char src
[INET_ADDRSTRLEN
], dst
[INET_ADDRSTRLEN
], next
[INET_ADDRSTRLEN
];
238 char src_if
[IFNAMSIZ
], dst_if
[IFNAMSIZ
];
242 if (recs
->nentries
== 0)
246 for (i
= 0; i
< recs
->nentries
; i
++, fle
++) {
247 inet_ntop(AF_INET
, &fle
->r
.r_src
, src
, sizeof(src
));
248 inet_ntop(AF_INET
, &fle
->r
.r_dst
, dst
, sizeof(dst
));
249 inet_ntop(AF_INET
, &fle
->next_hop
, next
, sizeof(next
));
250 printf(CISCO_SH_VERB_FLOW
,
251 if_indextoname(fle
->fle_i_ifx
, src_if
),
253 if_indextoname(fle
->fle_o_ifx
, dst_if
),
259 ntohs(fle
->r
.r_sport
),
262 ntohs(fle
->r
.r_dport
),
266 (u_int
)(fle
->bytes
/ fle
->packets
),
277 extern char *__progname
;
279 fprintf(stderr
, "usage: %s [-d level] nodename command\n", __progname
);