Compare commits
7 Commits
Author | SHA1 | Date | |
---|---|---|---|
f5c4e634cd | |||
3db023f13b | |||
a0f960c16f | |||
57a7653632 | |||
af508f0b4c | |||
b6267f7d0b | |||
0b00c0319c |
5
LICENSE
5
LICENSE
@ -1,6 +1,6 @@
|
|||||||
ISC License
|
ISC License
|
||||||
|
|
||||||
Copyright 2016-2022 Aaron Marcher <me@drkhsh.at>
|
Copyright 2016-2025 Aaron Marcher <me@drkhsh.at>
|
||||||
|
|
||||||
Copyright 2016 Roy Freytag <rfreytag@hs-mittweida.de>
|
Copyright 2016 Roy Freytag <rfreytag@hs-mittweida.de>
|
||||||
Copyright 2016 Vincent Loupmon <vincentloupmon@gmail.com>
|
Copyright 2016 Vincent Loupmon <vincentloupmon@gmail.com>
|
||||||
@ -29,6 +29,9 @@ Copyright 2020 Daniel Moch <daniel@danielmoch.com>
|
|||||||
Copyright 2022 Nickolas Raymond Kaczynski <nrk@disroot.org>
|
Copyright 2022 Nickolas Raymond Kaczynski <nrk@disroot.org>
|
||||||
Copyright 2022 Patrick Iacob <iacobp@oregonstate.edu>
|
Copyright 2022 Patrick Iacob <iacobp@oregonstate.edu>
|
||||||
Copyright 2021-2022 Steven Ward <planet36@gmail.com>
|
Copyright 2021-2022 Steven Ward <planet36@gmail.com>
|
||||||
|
Copyright 2025 Joakim Sindholt <opensource@zhasha.com>
|
||||||
|
Copyright 2025 Al <eirann@disroot.org>
|
||||||
|
Copyright 2025 sewn <sewn@disroot.org>
|
||||||
|
|
||||||
Permission to use, copy, modify, and/or distribute this software for any
|
Permission to use, copy, modify, and/or distribute this software for any
|
||||||
purpose with or without fee is hereby granted, provided that the above
|
purpose with or without fee is hereby granted, provided that the above
|
||||||
|
2
README
2
README
@ -18,7 +18,7 @@ Features
|
|||||||
- Available entropy
|
- Available entropy
|
||||||
- Username/GID/UID
|
- Username/GID/UID
|
||||||
- Hostname
|
- Hostname
|
||||||
- IP address (IPv4 and IPv6)
|
- IP address (IPv4 and IPv6), interface status
|
||||||
- Kernel version
|
- Kernel version
|
||||||
- Keyboard indicators
|
- Keyboard indicators
|
||||||
- Keymap
|
- Keymap
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
/* See LICENSE file for copyright and license details. */
|
/* See LICENSE file for copyright and license details. */
|
||||||
#include <ifaddrs.h>
|
#include <ifaddrs.h>
|
||||||
#include <netdb.h>
|
#include <netdb.h>
|
||||||
|
#include <net/if.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#if defined(__OpenBSD__)
|
#if defined(__OpenBSD__)
|
||||||
@ -59,3 +60,28 @@ ipv6(const char *interface)
|
|||||||
{
|
{
|
||||||
return ip(interface, AF_INET6);
|
return ip(interface, AF_INET6);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const char *
|
||||||
|
up(const char *interface)
|
||||||
|
{
|
||||||
|
struct ifaddrs *ifaddr, *ifa;
|
||||||
|
|
||||||
|
if (getifaddrs(&ifaddr) < 0) {
|
||||||
|
warn("getifaddrs:");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (ifa = ifaddr; ifa != NULL; ifa = ifa->ifa_next) {
|
||||||
|
if (!ifa->ifa_addr)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (!strcmp(ifa->ifa_name, interface)) {
|
||||||
|
freeifaddrs(ifaddr);
|
||||||
|
return ifa->ifa_flags & IFF_UP ? "up" : "down";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
freeifaddrs(ifaddr);
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
@ -29,8 +29,8 @@ get_layout(char *syms, int grp_num)
|
|||||||
int grp;
|
int grp;
|
||||||
|
|
||||||
layout = NULL;
|
layout = NULL;
|
||||||
tok = strtok(syms, "+:");
|
tok = strtok(syms, "+:_");
|
||||||
for (grp = 0; tok && grp <= grp_num; tok = strtok(NULL, "+:")) {
|
for (grp = 0; tok && grp <= grp_num; tok = strtok(NULL, "+:_")) {
|
||||||
if (!valid_layout_or_variant(tok)) {
|
if (!valid_layout_or_variant(tok)) {
|
||||||
continue;
|
continue;
|
||||||
} else if (strlen(tok) == 1 && isdigit(tok[0])) {
|
} else if (strlen(tok) == 1 && isdigit(tok[0])) {
|
||||||
|
@ -15,86 +15,232 @@
|
|||||||
(2 * (rssi + 100)))
|
(2 * (rssi + 100)))
|
||||||
|
|
||||||
#if defined(__linux__)
|
#if defined(__linux__)
|
||||||
#include <limits.h>
|
#include <stdint.h>
|
||||||
#include <linux/wireless.h>
|
#include <net/if.h>
|
||||||
|
#include <linux/netlink.h>
|
||||||
|
#include <linux/genetlink.h>
|
||||||
|
#include <linux/nl80211.h>
|
||||||
|
|
||||||
#define NET_OPERSTATE "/sys/class/net/%s/operstate"
|
static int nlsock = -1;
|
||||||
|
static uint32_t seq = 1;
|
||||||
|
static char resp[4096];
|
||||||
|
|
||||||
const char *
|
static char *
|
||||||
wifi_perc(const char *interface)
|
findattr(int attr, const char *p, const char *e, size_t *len)
|
||||||
{
|
{
|
||||||
int cur;
|
while (p < e) {
|
||||||
size_t i;
|
struct nlattr nla;
|
||||||
char *p, *datastart;
|
memcpy(&nla, p, sizeof(nla));
|
||||||
char path[PATH_MAX];
|
if (nla.nla_type == attr) {
|
||||||
char status[5];
|
*len = nla.nla_len - NLA_HDRLEN;
|
||||||
FILE *fp;
|
return (char *)(p + NLA_HDRLEN);
|
||||||
|
}
|
||||||
if (esnprintf(path, sizeof(path), NET_OPERSTATE, interface) < 0)
|
p += NLA_ALIGN(nla.nla_len);
|
||||||
return NULL;
|
|
||||||
if (!(fp = fopen(path, "r"))) {
|
|
||||||
warn("fopen '%s':", path);
|
|
||||||
return NULL;
|
|
||||||
}
|
}
|
||||||
p = fgets(status, 5, fp);
|
|
||||||
fclose(fp);
|
|
||||||
if (!p || strcmp(status, "up\n") != 0)
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
if (!(fp = fopen("/proc/net/wireless", "r"))) {
|
|
||||||
warn("fopen '/proc/net/wireless':");
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (i = 0; i < 3; i++)
|
static uint16_t
|
||||||
if (!(p = fgets(buf, sizeof(buf) - 1, fp)))
|
nl80211fam(void)
|
||||||
break;
|
{
|
||||||
|
static const char family[] = "nl80211";
|
||||||
|
static uint16_t id;
|
||||||
|
ssize_t r;
|
||||||
|
size_t len;
|
||||||
|
char ctrl[NLMSG_HDRLEN+GENL_HDRLEN+NLA_HDRLEN+NLA_ALIGN(sizeof(family))] = {0}, *p = ctrl;
|
||||||
|
|
||||||
fclose(fp);
|
if (id)
|
||||||
if (i < 2 || !p)
|
return id;
|
||||||
return NULL;
|
|
||||||
|
|
||||||
if (!(datastart = strstr(buf, interface)))
|
memcpy(p, &(struct nlmsghdr){
|
||||||
return NULL;
|
.nlmsg_len = sizeof(ctrl),
|
||||||
|
.nlmsg_type = GENL_ID_CTRL,
|
||||||
|
.nlmsg_flags = NLM_F_REQUEST,
|
||||||
|
.nlmsg_seq = seq++,
|
||||||
|
.nlmsg_pid = 0,
|
||||||
|
}, sizeof(struct nlmsghdr));
|
||||||
|
p += NLMSG_HDRLEN;
|
||||||
|
memcpy(p, &(struct genlmsghdr){
|
||||||
|
.cmd = CTRL_CMD_GETFAMILY,
|
||||||
|
.version = 1,
|
||||||
|
}, sizeof(struct genlmsghdr));
|
||||||
|
p += GENL_HDRLEN;
|
||||||
|
memcpy(p, &(struct nlattr){
|
||||||
|
.nla_len = NLA_HDRLEN+sizeof(family),
|
||||||
|
.nla_type = CTRL_ATTR_FAMILY_NAME,
|
||||||
|
}, sizeof(struct nlattr));
|
||||||
|
p += NLA_HDRLEN;
|
||||||
|
memcpy(p, family, sizeof(family));
|
||||||
|
|
||||||
datastart = (datastart+(strlen(interface)+1));
|
if (nlsock < 0)
|
||||||
sscanf(datastart + 1, " %*d %d %*d %*d\t\t %*d\t "
|
nlsock = socket(AF_NETLINK, SOCK_RAW, NETLINK_GENERIC);
|
||||||
"%*d\t\t%*d\t\t %*d\t %*d\t\t %*d", &cur);
|
if (nlsock < 0) {
|
||||||
|
warn("socket 'AF_NETLINK':");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (send(nlsock, ctrl, sizeof(ctrl), 0) != sizeof(ctrl)) {
|
||||||
|
warn("send 'AF_NETLINK':");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
r = recv(nlsock, resp, sizeof(resp), 0);
|
||||||
|
if (r < 0) {
|
||||||
|
warn("recv 'AF_NETLINK':");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if ((size_t)r <= sizeof(ctrl))
|
||||||
|
return 0;
|
||||||
|
p = findattr(CTRL_ATTR_FAMILY_ID, resp + sizeof(ctrl), resp + r, &len);
|
||||||
|
if (p && len == 2)
|
||||||
|
memcpy(&id, p, 2);
|
||||||
|
|
||||||
/* 70 is the max of /proc/net/wireless */
|
return id;
|
||||||
return bprintf("%d", (int)((float)cur / 70 * 100));
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
ifindex(const char *interface)
|
||||||
|
{
|
||||||
|
static struct ifreq ifr;
|
||||||
|
static int ifsock = -1;
|
||||||
|
|
||||||
|
if (ifsock < 0)
|
||||||
|
ifsock = socket(AF_UNIX, SOCK_DGRAM, 0);
|
||||||
|
if (ifsock < 0) {
|
||||||
|
warn("socket 'AF_UNIX':");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (strcmp(ifr.ifr_name, interface) != 0) {
|
||||||
|
strcpy(ifr.ifr_name, interface);
|
||||||
|
if (ioctl(ifsock, SIOCGIFINDEX, &ifr) != 0) {
|
||||||
|
warn("ioctl 'SIOCGIFINDEX':");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ifr.ifr_ifindex;
|
||||||
}
|
}
|
||||||
|
|
||||||
const char *
|
const char *
|
||||||
wifi_essid(const char *interface)
|
wifi_essid(const char *interface)
|
||||||
{
|
{
|
||||||
static char id[IW_ESSID_MAX_SIZE+1];
|
uint16_t fam = nl80211fam();
|
||||||
int sockfd;
|
ssize_t r;
|
||||||
struct iwreq wreq;
|
size_t len;
|
||||||
|
char req[NLMSG_HDRLEN+GENL_HDRLEN+NLA_HDRLEN+NLA_ALIGN(4)] = {0}, *p = req;
|
||||||
memset(&wreq, 0, sizeof(struct iwreq));
|
int idx = ifindex(interface);
|
||||||
wreq.u.essid.length = IW_ESSID_MAX_SIZE+1;
|
if (!fam) {
|
||||||
if (esnprintf(wreq.ifr_name, sizeof(wreq.ifr_name), "%s",
|
fprintf(stderr, "nl80211 family not found\n");
|
||||||
interface) < 0)
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
|
|
||||||
warn("socket 'AF_INET':");
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
wreq.u.essid.pointer = id;
|
if (idx < 0) {
|
||||||
if (ioctl(sockfd,SIOCGIWESSID, &wreq) < 0) {
|
fprintf(stderr, "interface %s not found\n", interface);
|
||||||
warn("ioctl 'SIOCGIWESSID':");
|
|
||||||
close(sockfd);
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
close(sockfd);
|
memcpy(p, &(struct nlmsghdr){
|
||||||
|
.nlmsg_len = sizeof(req),
|
||||||
|
.nlmsg_type = fam,
|
||||||
|
.nlmsg_flags = NLM_F_REQUEST,
|
||||||
|
.nlmsg_seq = seq++,
|
||||||
|
.nlmsg_pid = 0,
|
||||||
|
}, sizeof(struct nlmsghdr));
|
||||||
|
p += NLMSG_HDRLEN;
|
||||||
|
memcpy(p, &(struct genlmsghdr){
|
||||||
|
.cmd = NL80211_CMD_GET_INTERFACE,
|
||||||
|
.version = 1,
|
||||||
|
}, sizeof(struct genlmsghdr));
|
||||||
|
p += GENL_HDRLEN;
|
||||||
|
memcpy(p, &(struct nlattr){
|
||||||
|
.nla_len = NLA_HDRLEN+4,
|
||||||
|
.nla_type = NL80211_ATTR_IFINDEX,
|
||||||
|
}, sizeof(struct nlattr));
|
||||||
|
p += NLA_HDRLEN;
|
||||||
|
memcpy(p, &(uint32_t){idx}, 4);
|
||||||
|
|
||||||
if (!strcmp(id, ""))
|
if (send(nlsock, req, sizeof(req), 0) != sizeof(req)) {
|
||||||
|
warn("send 'AF_NETLINK':");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
r = recv(nlsock, resp, sizeof(resp), 0);
|
||||||
|
if (r < 0) {
|
||||||
|
warn("recv 'AF_NETLINK':");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((size_t)r <= NLMSG_HDRLEN + GENL_HDRLEN)
|
||||||
|
return NULL;
|
||||||
|
p = findattr(NL80211_ATTR_SSID, resp + NLMSG_HDRLEN + GENL_HDRLEN, resp + r, &len);
|
||||||
|
if (p)
|
||||||
|
p[len] = 0;
|
||||||
|
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *
|
||||||
|
wifi_perc(const char *interface)
|
||||||
|
{
|
||||||
|
static char strength[4];
|
||||||
|
struct nlmsghdr hdr;
|
||||||
|
uint16_t fam = nl80211fam();
|
||||||
|
ssize_t r;
|
||||||
|
size_t len;
|
||||||
|
char req[NLMSG_HDRLEN + GENL_HDRLEN + NLA_HDRLEN + NLA_ALIGN(4)] = {0}, *p = req, *e;
|
||||||
|
int idx = ifindex(interface);
|
||||||
|
|
||||||
|
if (idx < 0) {
|
||||||
|
fprintf(stderr, "interface %s not found\n", interface);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy(p, &(struct nlmsghdr){
|
||||||
|
.nlmsg_len = sizeof(req),
|
||||||
|
.nlmsg_type = fam,
|
||||||
|
.nlmsg_flags = NLM_F_REQUEST|NLM_F_DUMP,
|
||||||
|
.nlmsg_seq = seq++,
|
||||||
|
.nlmsg_pid = 0,
|
||||||
|
}, sizeof(struct nlmsghdr));
|
||||||
|
p += NLMSG_HDRLEN;
|
||||||
|
memcpy(p, &(struct genlmsghdr){
|
||||||
|
.cmd = NL80211_CMD_GET_STATION,
|
||||||
|
.version = 1,
|
||||||
|
}, sizeof(struct genlmsghdr));
|
||||||
|
p += GENL_HDRLEN;
|
||||||
|
memcpy(p, &(struct nlattr){
|
||||||
|
.nla_len = NLA_HDRLEN + 4,
|
||||||
|
.nla_type = NL80211_ATTR_IFINDEX,
|
||||||
|
}, sizeof(struct nlattr));
|
||||||
|
p += NLA_HDRLEN;
|
||||||
|
memcpy(p, &idx, 4);
|
||||||
|
|
||||||
|
if (send(nlsock, req, sizeof(req), 0) != sizeof(req)) {
|
||||||
|
warn("send 'AF_NETLINK':");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
*strength = 0;
|
||||||
|
while (1) {
|
||||||
|
r = recv(nlsock, resp, sizeof(resp), 0);
|
||||||
|
if (r < 0) {
|
||||||
|
warn("recv 'AF_NETLINK':");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
if ((size_t)r < sizeof(hdr))
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
return id;
|
for (p = resp; p != resp + r && (size_t)(resp + r-p) >= sizeof(hdr); p = e) {
|
||||||
|
memcpy(&hdr, p, sizeof(hdr));
|
||||||
|
e = resp + r - p < hdr.nlmsg_len ? resp + r : p + hdr.nlmsg_len;
|
||||||
|
|
||||||
|
if (!*strength && hdr.nlmsg_len > NLMSG_HDRLEN+GENL_HDRLEN) {
|
||||||
|
p += NLMSG_HDRLEN+GENL_HDRLEN;
|
||||||
|
p = findattr(NL80211_ATTR_STA_INFO, p, e, &len);
|
||||||
|
if (p)
|
||||||
|
p = findattr(NL80211_STA_INFO_SIGNAL_AVG, p, e, &len);
|
||||||
|
if (p && len == 1)
|
||||||
|
snprintf(strength, sizeof(strength), "%d", RSSI_TO_PERC(*p));
|
||||||
|
}
|
||||||
|
if (hdr.nlmsg_type == NLMSG_DONE)
|
||||||
|
return *strength ? strength : NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
#elif defined(__OpenBSD__)
|
#elif defined(__OpenBSD__)
|
||||||
#include <net/if.h>
|
#include <net/if.h>
|
||||||
|
@ -56,6 +56,7 @@ static const char unknown_str[] = "n/a";
|
|||||||
* thermal zone on FreeBSD
|
* thermal zone on FreeBSD
|
||||||
* (tz0, tz1, etc.)
|
* (tz0, tz1, etc.)
|
||||||
* uid UID of current user NULL
|
* uid UID of current user NULL
|
||||||
|
* up interface is running interface name (eth0)
|
||||||
* uptime system uptime NULL
|
* uptime system uptime NULL
|
||||||
* username username of current user NULL
|
* username username of current user NULL
|
||||||
* vol_perc OSS/ALSA volume in percent mixer file (/dev/mixer)
|
* vol_perc OSS/ALSA volume in percent mixer file (/dev/mixer)
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
# slstatus version
|
# slstatus version
|
||||||
VERSION = 1.0
|
VERSION = 1.1
|
||||||
|
|
||||||
# customize below to fit your system
|
# customize below to fit your system
|
||||||
|
|
||||||
|
@ -58,6 +58,7 @@ main(int argc, char *argv[])
|
|||||||
ARGBEGIN {
|
ARGBEGIN {
|
||||||
case 'v':
|
case 'v':
|
||||||
die("slstatus-"VERSION);
|
die("slstatus-"VERSION);
|
||||||
|
break;
|
||||||
case '1':
|
case '1':
|
||||||
done = 1;
|
done = 1;
|
||||||
/* FALLTHROUGH */
|
/* FALLTHROUGH */
|
||||||
|
@ -30,6 +30,7 @@ const char *hostname(const char *unused);
|
|||||||
/* ip */
|
/* ip */
|
||||||
const char *ipv4(const char *interface);
|
const char *ipv4(const char *interface);
|
||||||
const char *ipv6(const char *interface);
|
const char *ipv6(const char *interface);
|
||||||
|
const char *up(const char *interface);
|
||||||
|
|
||||||
/* kernel_release */
|
/* kernel_release */
|
||||||
const char *kernel_release(const char *unused);
|
const char *kernel_release(const char *unused);
|
||||||
|
Reference in New Issue
Block a user