altqstat.c revision 1.8 1 1.8 joerg /* $NetBSD: altqstat.c,v 1.8 2011/08/29 20:38:54 joerg Exp $ */
2 1.6 peter /* $KAME: altqstat.c,v 1.8 2002/10/27 03:19:35 kjc Exp $ */
3 1.1 thorpej /*
4 1.1 thorpej * Copyright (C) 1999-2000
5 1.1 thorpej * Sony Computer Science Laboratories, Inc. All rights reserved.
6 1.1 thorpej *
7 1.1 thorpej * Redistribution and use in source and binary forms, with or without
8 1.1 thorpej * modification, are permitted provided that the following conditions
9 1.1 thorpej * are met:
10 1.1 thorpej * 1. Redistributions of source code must retain the above copyright
11 1.1 thorpej * notice, this list of conditions and the following disclaimer.
12 1.1 thorpej * 2. Redistributions in binary form must reproduce the above copyright
13 1.1 thorpej * notice, this list of conditions and the following disclaimer in the
14 1.1 thorpej * documentation and/or other materials provided with the distribution.
15 1.1 thorpej *
16 1.1 thorpej * THIS SOFTWARE IS PROVIDED BY SONY CSL AND CONTRIBUTORS ``AS IS'' AND
17 1.1 thorpej * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18 1.1 thorpej * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19 1.1 thorpej * ARE DISCLAIMED. IN NO EVENT SHALL SONY CSL OR CONTRIBUTORS BE LIABLE
20 1.1 thorpej * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21 1.1 thorpej * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22 1.1 thorpej * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23 1.1 thorpej * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24 1.1 thorpej * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25 1.1 thorpej * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26 1.1 thorpej * SUCH DAMAGE.
27 1.1 thorpej */
28 1.1 thorpej
29 1.1 thorpej #include <sys/param.h>
30 1.1 thorpej #include <sys/time.h>
31 1.1 thorpej #include <sys/fcntl.h>
32 1.1 thorpej
33 1.1 thorpej #include <stdio.h>
34 1.1 thorpej #include <stdlib.h>
35 1.1 thorpej #include <unistd.h>
36 1.1 thorpej #include <string.h>
37 1.1 thorpej #include <signal.h>
38 1.1 thorpej #include <errno.h>
39 1.1 thorpej #include <err.h>
40 1.1 thorpej #ifndef NO_CURSES
41 1.1 thorpej #include <curses.h>
42 1.1 thorpej #endif
43 1.1 thorpej
44 1.1 thorpej #include "quip_client.h"
45 1.1 thorpej #include "altqstat.h"
46 1.1 thorpej
47 1.1 thorpej #define DEV_PATH "/dev/altq"
48 1.1 thorpej
49 1.1 thorpej int qdiscfd = -1;
50 1.1 thorpej int show_config = 0;
51 1.6 peter double ival = 5.0;
52 1.1 thorpej int no_server = 0;
53 1.1 thorpej char *interface = NULL;
54 1.1 thorpej char *qdisc_name = NULL;
55 1.1 thorpej
56 1.1 thorpej stat_loop_t *stat_loop;
57 1.1 thorpej
58 1.8 joerg __dead static void sig_handler(int);
59 1.6 peter static void alrm_handler(int);
60 1.8 joerg __dead static void usage(void);
61 1.1 thorpej
62 1.1 thorpej static void
63 1.1 thorpej sig_handler(int sig)
64 1.1 thorpej {
65 1.5 itojun char buf[8192];
66 1.5 itojun
67 1.5 itojun snprintf(buf, sizeof buf, "Exiting on signal %d\n", sig);
68 1.5 itojun write(STDERR_FILENO, buf, strlen(buf));
69 1.1 thorpej
70 1.1 thorpej #ifndef NO_CURSES
71 1.5 itojun /* XXX signal race */
72 1.1 thorpej if (qdisc_name != NULL && strcmp(qdisc_name, "wfq") == 0)
73 1.1 thorpej endwin(); /* wfqstat uses curses */
74 1.1 thorpej #endif
75 1.5 itojun _exit(0);
76 1.1 thorpej }
77 1.1 thorpej
78 1.6 peter static void
79 1.6 peter alrm_handler(int sig)
80 1.6 peter {
81 1.6 peter /* nothing */
82 1.6 peter }
83 1.6 peter
84 1.1 thorpej static void
85 1.1 thorpej usage(void)
86 1.1 thorpej {
87 1.1 thorpej fprintf(stderr, "usage: altqstat [-enrs] [-c count] [-w wait] [-i interface|-I input_interface]\n");
88 1.1 thorpej exit(1);
89 1.1 thorpej }
90 1.1 thorpej
91 1.1 thorpej int
92 1.1 thorpej main (int argc, char **argv)
93 1.1 thorpej {
94 1.1 thorpej int ch, raw_mode = 0;
95 1.6 peter int qtype, interval;
96 1.1 thorpej int count = 0;
97 1.6 peter struct itimerval it;
98 1.1 thorpej char device[64], qname[64], input[32];
99 1.1 thorpej
100 1.1 thorpej while ((ch = getopt(argc, argv, "I:c:ei:nrsw:")) != -1) {
101 1.1 thorpej switch (ch) {
102 1.1 thorpej case 'I':
103 1.2 itojun snprintf(input, sizeof(input), "_%s", optarg);
104 1.1 thorpej interface = input;
105 1.1 thorpej break;
106 1.1 thorpej case 'c':
107 1.1 thorpej count = atoi(optarg);
108 1.7 peter if (count < 1)
109 1.7 peter errx(1, "Please supply a count value bigger than 0.");
110 1.1 thorpej break;
111 1.1 thorpej case 'e':
112 1.1 thorpej quip_echo = 1;
113 1.1 thorpej break;
114 1.1 thorpej case 'i':
115 1.1 thorpej interface = optarg;
116 1.1 thorpej break;
117 1.1 thorpej case 'n':
118 1.1 thorpej no_server = 1;
119 1.1 thorpej break;
120 1.1 thorpej case 'r':
121 1.1 thorpej raw_mode = 1;
122 1.1 thorpej quip_echo = 1;
123 1.1 thorpej break;
124 1.1 thorpej case 's':
125 1.1 thorpej show_config = 1;
126 1.1 thorpej break;
127 1.1 thorpej case 'w':
128 1.6 peter ival = strtod(optarg, NULL);
129 1.1 thorpej break;
130 1.1 thorpej default:
131 1.1 thorpej usage();
132 1.1 thorpej break;
133 1.1 thorpej }
134 1.1 thorpej }
135 1.1 thorpej
136 1.1 thorpej signal(SIGINT, sig_handler);
137 1.1 thorpej signal(SIGTERM, sig_handler);
138 1.1 thorpej signal(SIGPIPE, sig_handler);
139 1.6 peter signal(SIGALRM, alrm_handler);
140 1.1 thorpej
141 1.1 thorpej if (no_server == 0) {
142 1.1 thorpej if (quip_openserver() < 0 && interface == NULL)
143 1.1 thorpej errx(1, "you have to specify interface!");
144 1.1 thorpej }
145 1.1 thorpej
146 1.1 thorpej if (raw_mode == 1) {
147 1.1 thorpej quip_rawmode();
148 1.1 thorpej quip_closeserver();
149 1.1 thorpej exit(0);
150 1.1 thorpej }
151 1.1 thorpej
152 1.1 thorpej if (show_config) {
153 1.1 thorpej if (no_server)
154 1.1 thorpej errx(1, "no server (-n) can't be set for show config (-s)!");
155 1.1 thorpej quip_printconfig();
156 1.1 thorpej quip_closeserver();
157 1.1 thorpej exit(0);
158 1.1 thorpej }
159 1.1 thorpej
160 1.1 thorpej interface = quip_selectinterface(interface);
161 1.1 thorpej if (interface == NULL)
162 1.1 thorpej errx(1, "no interface found!");
163 1.1 thorpej
164 1.1 thorpej qtype = ifname2qdisc(interface, qname);
165 1.1 thorpej if (qtype == 0)
166 1.1 thorpej errx(1, "altq is not attached on %s!", interface);
167 1.1 thorpej
168 1.1 thorpej qdisc_name = qname;
169 1.1 thorpej
170 1.1 thorpej stat_loop = qdisc2stat_loop(qdisc_name);
171 1.1 thorpej if (stat_loop == NULL)
172 1.1 thorpej errx(1, "qdisc %s is not supported!", qdisc_name);
173 1.1 thorpej
174 1.1 thorpej printf("%s: %s on interface %s\n",
175 1.1 thorpej argv[0], qdisc_name, interface);
176 1.1 thorpej
177 1.2 itojun snprintf(device, sizeof(device), "%s/%s", DEV_PATH, qdisc_name);
178 1.1 thorpej if ((qdiscfd = open(device, O_RDONLY)) < 0)
179 1.1 thorpej err(1, "can't open %s", device);
180 1.1 thorpej
181 1.6 peter interval = (int)(ival * 1000.0);
182 1.6 peter it.it_interval.tv_sec = interval / 1000;
183 1.6 peter it.it_interval.tv_usec = interval % 1000 * 1000;
184 1.6 peter it.it_value = it.it_interval;
185 1.6 peter setitimer(ITIMER_REAL, &it, NULL);
186 1.6 peter
187 1.6 peter (*stat_loop)(qdiscfd, interface, count, (int)ival);
188 1.1 thorpej
189 1.1 thorpej exit(0);
190 1.1 thorpej }
191 1.1 thorpej
192 1.1 thorpej /* calculate interval in sec */
193 1.1 thorpej double
194 1.1 thorpej calc_interval(struct timeval *cur_time, struct timeval *last_time)
195 1.1 thorpej {
196 1.1 thorpej double sec;
197 1.1 thorpej
198 1.1 thorpej sec = (double)(cur_time->tv_sec - last_time->tv_sec) +
199 1.1 thorpej (double)(cur_time->tv_usec - last_time->tv_usec) / 1000000;
200 1.1 thorpej return (sec);
201 1.1 thorpej }
202 1.1 thorpej
203 1.1 thorpej
204 1.1 thorpej /* calculate rate in bps */
205 1.1 thorpej double
206 1.1 thorpej calc_rate(u_int64_t new_bytes, u_int64_t last_bytes, double interval)
207 1.1 thorpej {
208 1.1 thorpej double rate;
209 1.1 thorpej
210 1.1 thorpej rate = (double)(new_bytes - last_bytes) * 8 / interval;
211 1.1 thorpej return (rate);
212 1.1 thorpej }
213 1.1 thorpej
214 1.1 thorpej /* calculate packets in second */
215 1.1 thorpej double
216 1.1 thorpej calc_pps(u_int64_t new_pkts, u_int64_t last_pkts, double interval)
217 1.1 thorpej {
218 1.1 thorpej double pps;
219 1.1 thorpej
220 1.1 thorpej pps = (double)(new_pkts - last_pkts) / interval;
221 1.1 thorpej return (pps);
222 1.1 thorpej }
223 1.1 thorpej
224 1.1 thorpej #define R2S_BUFS 8
225 1.2 itojun #define RATESTR_MAX 16
226 1.1 thorpej char *
227 1.1 thorpej rate2str(double rate)
228 1.1 thorpej {
229 1.1 thorpej char *buf;
230 1.2 itojun static char r2sbuf[R2S_BUFS][RATESTR_MAX]; /* ring bufer */
231 1.1 thorpej static int idx = 0;
232 1.1 thorpej
233 1.1 thorpej buf = r2sbuf[idx++];
234 1.1 thorpej if (idx == R2S_BUFS)
235 1.1 thorpej idx = 0;
236 1.1 thorpej
237 1.1 thorpej if (rate == 0.0)
238 1.2 itojun snprintf(buf, RATESTR_MAX, "0");
239 1.1 thorpej else if (rate >= 1000000.0)
240 1.2 itojun snprintf(buf, RATESTR_MAX, "%.2fM", rate / 1000000.0);
241 1.1 thorpej else
242 1.6 peter snprintf(buf, RATESTR_MAX, "%.2fk", rate / 1000.0);
243 1.1 thorpej return (buf);
244 1.1 thorpej }
245