4 Copyright (C) 2006-2009 Jonathan Zarate
11 #include <sys/sysinfo.h>
13 #include <sys/ioctl.h>
16 #include <sys/socket.h>
17 #include <netinet/in.h>
18 #include <arpa/inet.h>
19 #include <sys/types.h>
22 #include <sys/statfs.h>
24 #include <net/route.h>
33 // to javascript-safe string
34 char *js_string(const char *s
)
40 if ((buffer
= malloc((strlen(s
) * 4) + 1)) != NULL
) {
42 while ((c
= *s
++) != 0) {
43 if ((c
== '"') || (c
== '\'') || (c
== '\\') || (!isprint(c
))) {
44 b
+= sprintf(b
, "\\x%02x", c
);
55 // to html-safe string
56 char *html_string(const char *s
)
62 if ((buffer
= malloc((strlen(s
) * 6) + 1)) != NULL
) {
64 while ((c
= *s
++) != 0) {
65 if ((c
== '&') || (c
== '<') || (c
== '>') || (c
== '"') || (c
== '\'') || (!isprint(c
))) {
66 b
+= sprintf(b
, "&#%d;", c
);
78 char *unix_string(const char *s
)
84 if ((buffer
= malloc(strlen(s
) + 1)) != NULL
) {
86 while ((c
= *s
++) != 0)
87 if (c
!= '\r') *b
++ = c
;
94 char *reltime(char *buf
, time_t t
)
102 sprintf(buf
, "%d day%s, %02d:%02d:%02d", days
, ((days
==1) ? "" : "s"), ((m
/ 60) % 24), (m
% 60), (int)(t
% 60));
106 int get_client_info(char *mac
, char *ifname
)
111 char ip
[INET6_ADDRSTRLEN
];
113 char ip
[INET_ADDRSTRLEN
];
117 # ip neigh show fe80:0:0::201:02ff:fe03:0405
118 fe80::201:2ff:fe3:405 dev br0 lladdr 00:01:02:03:04:05 REACHABLE
120 if (clientsai
.ss_family
== AF_INET
) {
121 inet_ntop(clientsai
.ss_family
, &(((struct sockaddr_in
*)&clientsai
)->sin_addr
), ip
, sizeof(ip
));
122 sprintf(s
, "ip neigh show %s", ip
);
125 else if (clientsai
.ss_family
== AF_INET6
) {
126 inet_ntop(clientsai
.ss_family
, &(((struct sockaddr_in6
*)&clientsai
)->sin6_addr
), ip
, sizeof(ip
));
127 if (IN6_IS_ADDR_V4MAPPED( &(((struct sockaddr_in6
*)&clientsai
)->sin6_addr
) ))
128 sprintf(s
, "ip neigh show %s", ip
+ 7); // chop off the ::ffff: to get the ipv4 bit
130 sprintf(s
, "ip neigh show %s", ip
);
134 if ((f
= popen(s
, "r")) != NULL
) {
135 while (fgets(s
, sizeof(s
), f
)) {
136 if (sscanf(s
, "%*s dev %16s lladdr %17s %*s", ifname
, mac
) == 2) {
147 // <% lanip(mode); %>
149 // 1 return first 3 octets (192.168.1)
150 // 2 return last octet (1)
151 // else return full (192.168.1.1)
153 void asp_lanip(int argc
, char **argv
)
159 mode
= argc
? *argv
[0] : 0;
161 if ((nv
= nvram_get("lan_ipaddr")) != NULL
) {
163 if ((p
= strrchr(s
, '.')) != NULL
) {
165 web_puts((mode
== '1') ? s
: (mode
== '2') ? (p
+ 1) : nv
);
170 void asp_lipp(int argc
, char **argv
)
176 // <% psup(process); %>
177 // returns 1 if process is running
179 void asp_psup(int argc
, char **argv
)
181 if (argc
== 1) web_printf("%d", pidof(argv
[0]) > 0);
184 void wo_vpn_status(char *url
)
186 #ifdef TCONFIG_OPENVPN
194 if ( (str
= webcgi_get("server")) )
196 else if ( (str
= webcgi_get("client")) )
199 num
= str
? atoi(str
): 0;
200 if ( type
&& num
> 0 )
202 // Trigger OpenVPN to update the status file
203 snprintf(&buf
[0], sizeof(buf
), "vpn%s%d", type
, num
);
204 killall(&buf
[0], SIGUSR2
);
206 // Give it a chance to update the file
209 // Read the status file and repeat it verbatim to the caller
210 snprintf(&buf
[0], sizeof(buf
), "/etc/openvpn/%s%d/status", type
, num
);
211 fp
= fopen(&buf
[0], "r");
214 while (fgets(&buf
[0], sizeof(buf
), fp
) != NULL
)
224 total: used: free: shared: buffers: cached:
225 Mem: 14872576 12877824 1994752 0 1236992 4837376
247 unsigned long shared
;
248 unsigned long buffers
;
249 unsigned long cached
;
250 unsigned long swaptotal
;
251 unsigned long swapfree
;
252 unsigned long maxfreeram
;
255 static int get_memory(meminfo_t
*m
)
261 memset(m
, 0, sizeof(*m
));
262 if ((f
= fopen("/proc/meminfo", "r")) != NULL
) {
263 while (fgets(s
, sizeof(s
), f
)) {
265 if (strncmp(s
, "MemTotal:", 9) == 0) {
266 m
->total
= strtoul(s
+ 12, NULL
, 10) * 1024;
269 else if (strncmp(s
, "MemFree:", 8) == 0) {
270 m
->free
= strtoul(s
+ 12, NULL
, 10) * 1024;
273 else if (strncmp(s
, "Buffers:", 8) == 0) {
274 m
->buffers
= strtoul(s
+ 12, NULL
, 10) * 1024;
277 else if (strncmp(s
, "Cached:", 7) == 0) {
278 m
->cached
= strtoul(s
+ 12, NULL
, 10) * 1024;
282 if (strncmp(s
, "Mem:", 4) == 0) {
283 if (sscanf(s
+ 6, "%ld %*d %ld %ld %ld %ld", &m
->total
, &m
->free
, &m
->shared
, &m
->buffers
, &m
->cached
) == 5)
287 else if (strncmp(s
, "SwapTotal:", 10) == 0) {
288 m
->swaptotal
= strtoul(s
+ 12, NULL
, 10) * 1024;
291 else if (strncmp(s
, "SwapFree:", 9) == 0) {
292 m
->swapfree
= strtoul(s
+ 11, NULL
, 10) * 1024;
304 m
->maxfreeram
= m
->free
;
305 if (nvram_match("t_cafree", "1")) m
->maxfreeram
+= (m
->cached
+ m
->buffers
);
310 #define IP6ADDR_MAX_CNT 3 // wan, lan, lan-ll
311 static void print_ipv6_addrs(void)
313 char buf
[INET6_ADDRSTRLEN
];
316 struct ifaddrs
*ifap
, *ifa
;
317 struct sockaddr_in6
*s6
;
319 if (!ipv6_enabled() || (getifaddrs(&ifap
) != 0))
323 for (ifa
= ifap
; ifa
; ifa
= ifa
->ifa_next
) {
324 if ((ifa
->ifa_addr
== NULL
) || (ifa
->ifa_addr
->sa_family
!= AF_INET6
))
327 s6
= (struct sockaddr_in6
*)(ifa
->ifa_addr
);
329 if (strncmp(ifa
->ifa_name
, nvram_safe_get("lan_ifname"), IFNAMSIZ
) == 0) {
330 if (IN6_IS_ADDR_LINKLOCAL(&s6
->sin6_addr
))
335 else if (strncmp(ifa
->ifa_name
, get_wan6face(), IFNAMSIZ
) == 0) {
336 if (!IN6_IS_ADDR_LINKLOCAL(&s6
->sin6_addr
))
344 if (inet_ntop(ifa
->ifa_addr
->sa_family
, &(s6
->sin6_addr
), buf
, sizeof(buf
)) != NULL
) {
345 web_printf("\tip6_%s: '%s',\n",
347 if (++found
>= IP6ADDR_MAX_CNT
)
355 void asp_calc6rdlocalprefix(int argc
, char **argv
)
357 struct in6_addr prefix_addr
, local_prefix_addr
;
358 int prefix_len
= 0, relay_prefix_len
= 0, local_prefix_len
= 0;
359 struct in_addr wanip_addr
;
360 char local_prefix
[INET6_ADDRSTRLEN
];
363 if (argc
!= 3) return;
365 inet_pton(AF_INET6
, argv
[0], &prefix_addr
);
366 prefix_len
= atoi(argv
[1]);
367 relay_prefix_len
= atoi(argv
[2]);
368 inet_pton(AF_INET
, get_wanip(), &wanip_addr
);
370 if (calc_6rd_local_prefix(&prefix_addr
, prefix_len
, relay_prefix_len
,
371 &wanip_addr
, &local_prefix_addr
, &local_prefix_len
) &&
372 inet_ntop(AF_INET6
, &local_prefix_addr
, local_prefix
, sizeof(local_prefix
)) != NULL
) {
373 sprintf(s
, "\nlocal_prefix = '%s/%d';\n", local_prefix
, local_prefix_len
);
383 dev: size erasesize name
384 mtd0: 00020000 00010000 "pmon"
385 mtd1: 007d0000 00010000 "linux"
393 if ((f
= fopen("/proc/mtd", "r")) != NULL
) {
394 while (fgets(s
, sizeof(s
), f
)) {
395 if (sscanf(s
, "%*s %X %*s %16s", &size
, partname
) != 2) continue;
396 if (strcmp(partname
, "\"linux\"") == 0) {
404 if ((size
> 0x2000000) && (size
< 0x4000000)) return 64;
405 else if ((size
> 0x1000000) && (size
< 0x2000000)) return 32;
406 else if ((size
> 0x800000) && (size
< 0x1000000)) return 16;
407 else if ((size
> 0x400000) && (size
< 0x800000)) return 8;
408 else if ((size
> 0x200000) && (size
< 0x400000)) return 4;
409 else if ((size
> 0x100000) && (size
< 0x200000)) return 2;
417 void asp_etherstates(int argc
, char **argv
)
420 char s
[32], *a
, b
[16];
423 if (nvram_match("lan_state", "1")) {
425 web_puts("\netherstates = {");
427 system("/usr/sbin/ethstate");
429 if ((f
= fopen("/tmp/ethernet.state", "r")) != NULL
) {
430 while (fgets(s
, sizeof(s
), f
)) {
431 if (sscanf(s
, "Port 0: %s", b
) == 1) a
="port0";
432 else if (sscanf(s
, "Port 1: %s", b
) == 1) a
="port1";
433 else if (sscanf(s
, "Port 2: %s", b
) == 1) a
="port2";
434 else if (sscanf(s
, "Port 3: %s", b
) == 1) a
="port3";
435 else if (sscanf(s
, "Port 4: %s", b
) == 1) a
="port4";
438 web_printf("%s\t%s: '%s'", n
? ",\n" : "", a
, b
);
445 web_puts("\netherstates = {\tport0: 'disabled'\n};\n");
449 void asp_anonupdate(int argc
, char **argv
)
452 char s
[32], *a
, b
[16];
455 if ( nvram_match("tomatoanon_answer", "1") && nvram_match("tomatoanon_enable", "1") && nvram_match("tomatoanon_notify", "1") ) {
457 web_puts("\nanonupdate = {");
460 if ((f
= fopen("/tmp/anon.version", "r")) != NULL
) {
461 while (fgets(s
, sizeof(s
), f
)) {
462 if (sscanf(s
, "have_update=%s", b
) == 1) a
="update";
465 web_printf("%s\t%s: '%s'", n
? ",\n" : "", a
, b
);
472 web_puts("\nanonupdate = {\tupdate: 'no'\n};\n");
476 void asp_sysinfo(int argc
, char **argv
)
482 char system_type
[64];
487 get_cpuinfo(system_type
, cpu_model
, bogomips
, cpuclk
);
489 web_puts("\nsysinfo = {\n");
498 "\tuptime_s: '%s',\n"
499 "\tloads: [%ld, %ld, %ld],\n"
503 "\tbufferram: %ld,\n"
505 "\ttotalswap: %ld,\n"
507 "\ttotalfreeram: %ld,\n"
510 "\tsystemtype: '%s',\n"
511 "\tcpumodel: '%s',\n"
512 "\tbogomips: '%s',\n"
515 reltime(s
, si
.uptime
),
516 si
.loads
[0], si
.loads
[1], si
.loads
[2],
518 mem
.shared
, mem
.buffers
, mem
.cached
,
519 mem
.swaptotal
, mem
.swapfree
,
531 void asp_activeroutes(int argc
, char **argv
)
537 unsigned long gateway
;
547 web_puts("\nactiveroutes = [");
549 if ((f
= fopen("/proc/net/route", "r")) != NULL
) {
550 while (fgets(s
, sizeof(s
), f
)) {
551 if (sscanf(s
, "%16s%lx%lx%lx%*s%*s%u%lx", dev
, &dest
, &gateway
, &flags
, &metric
, &mask
) != 6) continue;
552 if ((flags
& RTF_UP
) == 0) continue;
555 strcpy(s_dest
, inet_ntoa(ia
));
558 strcpy(s_dest
, "default");
562 strcpy(s_gateway
, inet_ntoa(ia
));
565 strcpy(s_gateway
, "*");
568 strcpy(s_mask
, inet_ntoa(ia
));
569 web_printf("%s['%s','%s','%s','%s',%u]", n
? "," : "", dev
, s_dest
, s_gateway
, s_mask
, metric
);
578 struct sockaddr_in6 snaddr6
;
579 char addr6
[40], nhop6
[40];
581 if ((ipv6_enabled()) &&
582 (f
= fopen("/proc/net/ipv6_route", "r")) != NULL
) {
583 while (fgets(s
, sizeof(s
), f
)) {
584 if (sscanf(s
, "%32s%x%*s%*s%32s%x%*s%*s%lx%s\n",
585 addr6x
+14, &pxlen
, addr6x
+40+7, &metric
, &flags
, dev
) != 6) continue;
587 if ((flags
& RTF_UP
) == 0) continue;
593 if (i
== 40) { // nul terminator for 1st address?
594 addr6x
[39] = 0; // Fixup... need 0 instead of ':'.
595 ++p
; // Skip and continue.
604 } while (i
< 40+28+7);
606 inet_pton(AF_INET6
, addr6x
, (struct sockaddr
*) &snaddr6
.sin6_addr
);
607 if (IN6_IS_ADDR_UNSPECIFIED(&snaddr6
.sin6_addr
))
608 strcpy(addr6
, "default");
610 inet_ntop(AF_INET6
, &snaddr6
.sin6_addr
, addr6
, sizeof(addr6
));
612 inet_pton(AF_INET6
, addr6x
+ 40, (struct sockaddr
*) &snaddr6
.sin6_addr
);
613 if (IN6_IS_ADDR_UNSPECIFIED(&snaddr6
.sin6_addr
))
616 inet_ntop(AF_INET6
, &snaddr6
.sin6_addr
, nhop6
, sizeof(nhop6
));
618 web_printf("%s['%s','%s','%s','%d',%u]", n
? "," : "", dev
, addr6
, nhop6
, pxlen
, metric
);
629 void asp_cgi_get(int argc
, char **argv
)
634 for (i
= 0; i
< argc
; ++i
) {
635 v
= webcgi_get(argv
[i
]);
640 void asp_time(int argc
, char **argv
)
647 web_puts("Not Available");
650 strftime(s
, sizeof(s
), "%a, %d %b %Y %H:%M:%S %z", localtime(&t
));
656 void asp_mmcid(int argc
, char **argv
) {
658 char s
[32], *a
, b
[16];
661 web_puts("\nmmcid = {");
663 if ((f
= fopen("/proc/mmc/status", "r")) != NULL
) {
664 while (fgets(s
, sizeof(s
), f
)) {
666 if (sscanf(s
, "Card Type : %16s", b
) == 1) a
="type";
667 else if (sscanf(s
, "Spec Version : %16s", b
) == 1) a
="spec";
668 else if (sscanf(s
, "Num. of Blocks : %d", &size
) == 1) { a
="size"; snprintf(b
,sizeof(b
),"%lld",((unsigned long long)size
)*512); }
669 else if (sscanf(s
, "Voltage Range : %16s", b
) == 1) a
="volt";
670 else if (sscanf(s
, "Manufacture ID : %16s", b
) == 1) a
="manuf";
671 else if (sscanf(s
, "Application ID : %16s", b
) == 1) a
="appl";
672 else if (sscanf(s
, "Product Name : %16s", b
) == 1) a
="prod";
673 else if (sscanf(s
, "Revision : %16s", b
) == 1) a
="rev";
674 else if (sscanf(s
, "Serial Number : %16s", b
) == 1) a
="serial";
675 else if (sscanf(s
, "Manu. Date : %8c", b
) == 1) { a
="date"; b
[9]=0; }
677 web_printf(size
==1 ? "%s\t%s: '%s'" : "%s\t%s: %s", n
? ",\n" : "", a
, b
);
686 void asp_wanup(int argc
, char **argv
)
688 web_puts(check_wanup() ? "1" : "0");
691 void asp_wanstatus(int argc
, char **argv
)
695 if ((using_dhcpc()) && (f_exists("/var/lib/misc/dhcpc.renewing"))) {
698 else if (check_wanup()) {
701 else if (f_exists("/var/lib/misc/wan.connecting")) {
710 void asp_link_uptime(int argc
, char **argv
)
720 if (f_read("/var/lib/misc/wantime", &uptime
, sizeof(uptime
)) == sizeof(uptime
)) {
721 reltime(buf
, si
.uptime
- uptime
);
727 void asp_rrule(int argc
, char **argv
)
732 i
= nvram_get_int("rruleN");
733 sprintf(s
, "rrule%d", i
);
734 web_puts("\nrrule = '");
735 web_putj(nvram_safe_get(s
));
736 web_printf("';\nrruleN = %d;\n", i
);
739 void asp_compmac(int argc
, char **argv
)
744 if (get_client_info(mac
, ifname
)) {
749 void asp_ident(int argc
, char **argv
)
751 web_puth(nvram_safe_get("router_name"));
754 void asp_statfs(int argc
, char **argv
)
759 if (argc
!= 2) return;
761 // used for /cifs/, /jffs/... if it returns squashfs type, assume it's not mounted
762 if ((statfs(argv
[0], &sf
) != 0) || (sf
.f_type
== 0x73717368)) {
764 memset(&sf
, 0, sizeof(sf
));
766 // for jffs, try to get total size from mtd partition
767 if (strncmp(argv
[1], "jffs", 4) == 0) {
770 if (mtd_getinfo(argv
[1], &part
, (int *)&sf
.f_blocks
)) {
787 ((uint64_t)sf
.f_bsize
* sf
.f_blocks
),
788 ((uint64_t)sf
.f_bsize
* sf
.f_bfree
));
791 void asp_notice(int argc
, char **argv
)
796 if (argc
!= 1) return;
797 snprintf(s
, sizeof(s
), "/var/notice/%s", argv
[0]);
798 if (f_read_string(s
, buf
, sizeof(buf
)) <= 0) return;
802 void wo_wakeup(char *url
)
808 if ((mac
= webcgi_get("mac")) != NULL
) {
809 end
= mac
+ strlen(mac
);
811 while ((*mac
== ' ') || (*mac
== '\t') || (*mac
== '\r') || (*mac
== '\n')) ++mac
;
812 if (*mac
== 0) break;
815 while ((*p
!= 0) && (*p
!= ' ') && (*p
!= '\r') && (*p
!= '\n')) ++p
;
818 eval("ether-wake", "-b", "-i", nvram_safe_get("lan_ifname"), mac
);
819 if (strcmp(nvram_safe_get("lan1_ifname"), "") != 0)
820 eval("ether-wake", "-b", "-i", nvram_safe_get("lan1_ifname"), mac
);
821 if (strcmp(nvram_safe_get("lan2_ifname"), "") != 0)
822 eval("ether-wake", "-b", "-i", nvram_safe_get("lan2_ifname"), mac
);
823 if (strcmp(nvram_safe_get("lan3_ifname"), "") != 0)
824 eval("ether-wake", "-b", "-i", nvram_safe_get("lan3_ifname"), mac
);
831 void asp_dns(int argc
, char **argv
)
835 const dns_list_t
*dns
;
837 dns
= get_dns(); // static buffer
838 strcpy(s
, "\ndns = [");
839 for (i
= 0 ; i
< dns
->count
; ++i
) {
840 sprintf(s
+ strlen(s
), "%s'%s:%u'", i
? "," : "", inet_ntoa(dns
->dns
[i
].addr
), dns
->dns
[i
].port
);
846 int resolve_addr(const char *ip
, char *host
)
848 struct addrinfo hints
;
849 struct addrinfo
*res
;
852 memset(&hints
, 0, sizeof(hints
));
853 hints
.ai_family
= AF_UNSPEC
;
854 hints
.ai_socktype
= SOCK_STREAM
;
856 ret
= getaddrinfo(ip
, NULL
, &hints
, &res
);
858 ret
= getnameinfo(res
->ai_addr
, res
->ai_addrlen
, host
, NI_MAXHOST
, NULL
, 0, 0);
864 void wo_resolve(char *url
)
868 char host
[NI_MAXHOST
];
873 web_puts("\nresolve_data = [\n");
874 if ((p
= webcgi_get("ip")) != NULL
) {
875 while ((ip
= strsep(&p
, ",")) != NULL
) {
876 if (resolve_addr(ip
, host
) != 0) continue;
877 js
= js_string(host
);
878 web_printf("%c['%s','%s']", comma
, ip
, js
);