My slstatus configuration
Du kan inte välja fler än 25 ämnen Ämnen måste starta med en bokstav eller siffra, kan innehålla bindestreck ('-') och vara max 35 tecken långa.
 
 
 
 

175 rader
3.7 KiB

  1. /* See LICENSE file for copyright and license details. */
  2. #include <ifaddrs.h>
  3. #include <stdio.h>
  4. #include <string.h>
  5. #include <sys/ioctl.h>
  6. #include <sys/socket.h>
  7. #include <unistd.h>
  8. #include "../util.h"
  9. #if defined(__linux__)
  10. #include <limits.h>
  11. #include <linux/wireless.h>
  12. const char *
  13. wifi_perc(const char *interface)
  14. {
  15. int cur;
  16. size_t i;
  17. char *p, *datastart;
  18. char path[PATH_MAX];
  19. char status[5];
  20. FILE *fp;
  21. if (esnprintf(path, sizeof(path), "/sys/class/net/%s/operstate",
  22. interface) < 0) {
  23. return NULL;
  24. }
  25. if (!(fp = fopen(path, "r"))) {
  26. warn("fopen '%s':", path);
  27. return NULL;
  28. }
  29. p = fgets(status, 5, fp);
  30. fclose(fp);
  31. if(!p || strcmp(status, "up\n") != 0) {
  32. return NULL;
  33. }
  34. if (!(fp = fopen("/proc/net/wireless", "r"))) {
  35. warn("fopen '/proc/net/wireless':");
  36. return NULL;
  37. }
  38. for (i = 0; i < 3; i++) {
  39. if (!(p = fgets(buf, sizeof(buf) - 1, fp)))
  40. break;
  41. }
  42. fclose(fp);
  43. if (i < 2 || !p) {
  44. return NULL;
  45. }
  46. if (!(datastart = strstr(buf, interface))) {
  47. return NULL;
  48. }
  49. datastart = (datastart+(strlen(interface)+1));
  50. sscanf(datastart + 1, " %*d %d %*d %*d\t\t %*d\t "
  51. "%*d\t\t%*d\t\t %*d\t %*d\t\t %*d", &cur);
  52. /* 70 is the max of /proc/net/wireless */
  53. return bprintf("%d", (int)((float)cur / 70 * 100));
  54. }
  55. const char *
  56. wifi_essid(const char *interface)
  57. {
  58. static char id[IW_ESSID_MAX_SIZE+1];
  59. int sockfd;
  60. struct iwreq wreq;
  61. memset(&wreq, 0, sizeof(struct iwreq));
  62. wreq.u.essid.length = IW_ESSID_MAX_SIZE+1;
  63. if (esnprintf(wreq.ifr_name, sizeof(wreq.ifr_name), "%s",
  64. interface) < 0) {
  65. return NULL;
  66. }
  67. if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
  68. warn("socket 'AF_INET':");
  69. return NULL;
  70. }
  71. wreq.u.essid.pointer = id;
  72. if (ioctl(sockfd,SIOCGIWESSID, &wreq) < 0) {
  73. warn("ioctl 'SIOCGIWESSID':");
  74. close(sockfd);
  75. return NULL;
  76. }
  77. close(sockfd);
  78. if (!strcmp(id, "")) {
  79. return NULL;
  80. }
  81. return id;
  82. }
  83. #elif defined(__OpenBSD__)
  84. #include <net/if.h>
  85. #include <net/if_media.h>
  86. #include <net80211/ieee80211.h>
  87. #include <sys/select.h> /* before <sys/ieee80211_ioctl.h> for NBBY */
  88. #include <net80211/ieee80211_ioctl.h>
  89. #include <stdlib.h>
  90. #include <sys/types.h>
  91. static int
  92. load_ieee80211_nodereq(const char *interface, struct ieee80211_nodereq *nr)
  93. {
  94. struct ieee80211_bssid bssid;
  95. int sockfd;
  96. uint8_t zero_bssid[IEEE80211_ADDR_LEN];
  97. memset(&bssid, 0, sizeof(bssid));
  98. memset(nr, 0, sizeof(struct ieee80211_nodereq));
  99. if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
  100. warn("socket 'AF_INET':");
  101. return 0;
  102. }
  103. strlcpy(bssid.i_name, interface, sizeof(bssid.i_name));
  104. if ((ioctl(sockfd, SIOCG80211BSSID, &bssid)) < 0) {
  105. warn("ioctl 'SIOCG80211BSSID':");
  106. close(sockfd);
  107. return 0;
  108. }
  109. memset(&zero_bssid, 0, sizeof(zero_bssid));
  110. if (memcmp(bssid.i_bssid, zero_bssid,
  111. IEEE80211_ADDR_LEN) == 0) {
  112. close(sockfd);
  113. return 0;
  114. }
  115. strlcpy(nr->nr_ifname, interface, sizeof(nr->nr_ifname));
  116. memcpy(&nr->nr_macaddr, bssid.i_bssid, sizeof(nr->nr_macaddr));
  117. if ((ioctl(sockfd, SIOCG80211NODE, nr)) < 0 && nr->nr_rssi) {
  118. warn("ioctl 'SIOCG80211NODE':");
  119. close(sockfd);
  120. return 0;
  121. }
  122. return close(sockfd), 1;
  123. }
  124. const char *
  125. wifi_perc(const char *interface)
  126. {
  127. struct ieee80211_nodereq nr;
  128. int q;
  129. if (load_ieee80211_nodereq(interface, &nr)) {
  130. if (nr.nr_max_rssi) {
  131. q = IEEE80211_NODEREQ_RSSI(&nr);
  132. } else {
  133. q = nr.nr_rssi >= -50 ? 100 :
  134. (nr.nr_rssi <= -100 ? 0 :
  135. (2 * (nr.nr_rssi + 100)));
  136. }
  137. return bprintf("%d", q);
  138. }
  139. return NULL;
  140. }
  141. const char *
  142. wifi_essid(const char *interface)
  143. {
  144. struct ieee80211_nodereq nr;
  145. if (load_ieee80211_nodereq(interface, &nr)) {
  146. return bprintf("%s", nr.nr_nwid);
  147. }
  148. return NULL;
  149. }
  150. #endif