From 5a3b7c074b581957cd6606128a6d3207a37cfebe Mon Sep 17 00:00:00 2001 From: Phil Sutter Date: Sun, 6 Sep 2009 21:53:53 +0200 Subject: [PATCH] Rewrite hddtemp support for better scaling Instead of connecting once for each object, have a central update routine (limiting support to only a single hddtemp daemon to connect to). --- ChangeLog | 3 + doc/config_settings.xml | 20 +++ doc/variables.xml | 9 +- src/common.c | 5 + src/conky.c | 35 ++--- src/conky.h | 3 + src/core.c | 21 ++- src/hddtemp.c | 310 +++++++++++++++++++++++----------------- src/hddtemp.h | 7 +- 9 files changed, 240 insertions(+), 173 deletions(-) diff --git a/ChangeLog b/ChangeLog index 2c015465..bac63c66 100644 --- a/ChangeLog +++ b/ChangeLog @@ -1,3 +1,6 @@ +2009-09-06 + * Rewrite hddtemp support for better scaling. + 2009-08-10 * Add day and date data_type to $weather_forecast diff --git a/doc/config_settings.xml b/doc/config_settings.xml index 5414211a..1b71f439 100644 --- a/doc/config_settings.xml +++ b/doc/config_settings.xml @@ -276,6 +276,26 @@ For other position related stuff, see 'alignment'. + + + + + + + Hostname to connect to for hddtemp objects. Defaults + to "127.0.0.1". + + + + + + + + + Port to use for hddtemp connections. Defaults to + 7634. + + diff --git a/doc/variables.xml b/doc/variables.xml index 661ce257..ad8b856b 100644 --- a/doc/variables.xml +++ b/doc/variables.xml @@ -1251,12 +1251,13 @@ - + Displays temperature of a selected hard disk - drive as reported by the hddtemp daemon running on - host:port. Default host is 127.0.0.1, default port is 7634. - + drive as reported by the hddtemp daemon. Use hddtemp_host + and hddtemp_port to specify a host and port for all hddtemp + objects. If no dev parameter is given, the first disk returned + by the hddtemp daemon is used. diff --git a/src/common.c b/src/common.c index 1d142150..f8f4821b 100644 --- a/src/common.c +++ b/src/common.c @@ -530,6 +530,11 @@ void update_stuff(void) update_apcupsd(); } #endif +#ifdef HDDTEMP + if (NEED(INFO_HDDTEMP)) { + update_hddtemp(); + } +#endif } /* Ohkie to return negative values for temperatures */ diff --git a/src/conky.c b/src/conky.c index 40cb317b..dab97438 100644 --- a/src/conky.c +++ b/src/conky.c @@ -1966,29 +1966,14 @@ static void generate_text_internal(char *p, int p_max_size, #endif /* HAVE_LUA */ #ifdef HDDTEMP OBJ(hddtemp) { - char *endptr, unit; - long val; - if (obj->data.hddtemp.update_time < current_update_time - 30) { - if (obj->data.hddtemp.temp) - free(obj->data.hddtemp.temp); - obj->data.hddtemp.temp = get_hddtemp_info(obj->data.hddtemp.dev, - obj->data.hddtemp.addr, obj->data.hddtemp.port); - obj->data.hddtemp.update_time = current_update_time; - } - if (!obj->data.hddtemp.temp) { + short val; + char unit; + + if (get_hddtemp_info(obj->data.s, &val, &unit)) { snprintf(p, p_max_size, "N/A"); } else { - val = strtol(obj->data.hddtemp.temp + 1, &endptr, 10); - unit = obj->data.hddtemp.temp[0]; - - if (*endptr != '\0') - snprintf(p, p_max_size, "N/A"); - else if (unit == 'C') - temp_print(p, p_max_size, (double)val, TEMP_CELSIUS); - else if (unit == 'F') - temp_print(p, p_max_size, (double)val, TEMP_FAHRENHEIT); - else - snprintf(p, p_max_size, "N/A"); + temp_print(p, p_max_size, (double)val, + (unit == 'C' ? TEMP_CELSIUS : TEMP_FAHRENHEIT)); } } #endif @@ -5798,6 +5783,14 @@ char load_config_file(const char *f) CONF("format_human_readable") { format_human_readable = string_to_bool(value); } +#ifdef HDDTEMP + CONF("hddtemp_host") { + set_hddtemp_host(value); + } + CONF("hddtemp_port") { + set_hddtemp_port(value); + } +#endif /* HDDTEMP */ CONF("pad_percents") { pad_percents = atoi(value); } diff --git a/src/conky.h b/src/conky.h index 84288992..c8dd93b8 100644 --- a/src/conky.h +++ b/src/conky.h @@ -262,6 +262,9 @@ enum { #ifdef WEATHER INFO_WEATHER = 33, #endif +#ifdef HDDTEMP + INFO_HDDTEMP = 34, +#endif }; /* get_battery_stuff() item selector diff --git a/src/core.c b/src/core.c index b21bcf7e..bfcea6ee 100644 --- a/src/core.c +++ b/src/core.c @@ -1989,15 +1989,9 @@ struct text_object *construct_text_object(const char *s, const char *arg, long #endif /* X11 */ #endif /* HAVE_LUA */ #ifdef HDDTEMP - END OBJ(hddtemp, 0) - if (scan_hddtemp(arg, &obj->data.hddtemp.dev, - &obj->data.hddtemp.addr, &obj->data.hddtemp.port)) { - NORM_ERR("hddtemp needs arguments"); - obj->type = OBJ_text; - obj->data.s = strndup("${hddtemp}", text_buffer_size); - obj->data.hddtemp.update_time = 0; - } else - obj->data.hddtemp.temp = NULL; + END OBJ(hddtemp, INFO_HDDTEMP) + if (arg) + obj->data.s = strndup(arg, text_buffer_size); #endif /* HDDTEMP */ #ifdef TCP_PORT_MONITOR END OBJ(tcp_portmon, INFO_TCP_PORT_MONITOR) @@ -2672,10 +2666,11 @@ void free_text_objects(struct text_object *root, int internal) break; #ifdef HDDTEMP case OBJ_hddtemp: - free(data.hddtemp.dev); - free(data.hddtemp.addr); - if (data.hddtemp.temp) - free(data.hddtemp.temp); + if (data.s) { + free(data.s); + data.s = NULL; + } + free_hddtemp(); break; #endif /* HDDTEMP */ case OBJ_entropy_avail: diff --git a/src/hddtemp.c b/src/hddtemp.c index d053305b..57001d4b 100644 --- a/src/hddtemp.c +++ b/src/hddtemp.c @@ -40,173 +40,217 @@ #include #define BUFLEN 512 -#define PORT 7634 +#define DEFAULT_PORT "7634" +#define DEFAULT_HOST "127.0.0.1" -char buf[BUFLEN]; +static char *hddtemp_host = NULL; +static char *hddtemp_port = NULL; -int scan_hddtemp(const char *arg, char **dev, char **addr, int *port) +static struct hdd_info { + struct hdd_info *next; + char *dev; + short temp; + char unit; +} hdd_info_head = { + .next = NULL, +}; + +void set_hddtemp_host(const char *host) { - char buf1[32], buf2[64]; - int n, ret; + if (hddtemp_host) + free(hddtemp_host); + hddtemp_host = strdup(host); +} - if (!arg) - return 1; +void set_hddtemp_port(const char *port) +{ + if (hddtemp_port) + free(hddtemp_port); + hddtemp_port = strdup(port); +} - if ((ret = sscanf(arg, "%31s %63s %d", buf1, buf2, &n)) < 1) - return 1; +static void __free_hddtemp_info(struct hdd_info *hdi) +{ + if (hdi->next) + __free_hddtemp_info(hdi->next); + free(hdi->dev); + free(hdi); +} - if (strncmp(buf1, "/dev/", 5)) { - strncpy(buf1 + 5, buf1, 32 - 5); - strncpy(buf1, "/dev/", 5); - } - *dev = strndup(buf1, text_buffer_size); +static void free_hddtemp_info(void) +{ + DBGP("free_hddtemp_info() called"); + if (!hdd_info_head.next) + return; + __free_hddtemp_info(hdd_info_head.next); + hdd_info_head.next = NULL; +} - if (ret >= 2) { - *addr = strndup(buf2, text_buffer_size); - } else { - *addr = strndup("127.0.0.1", text_buffer_size); +static void add_hddtemp_info(char *dev, short temp, char unit) +{ + struct hdd_info *hdi = &hdd_info_head; + + DBGP("add_hddtemp_info(%s, %d, %c) being called", dev, temp, unit); + while (hdi->next) + hdi = hdi->next; + + hdi->next = malloc(sizeof(struct hdd_info)); + memset(hdi->next, 0, sizeof(struct hdd_info)); + hdi->next->dev = strdup(dev); + hdi->next->temp = temp; + hdi->next->unit = unit; +} + +static char *fetch_hddtemp_output(void) +{ + int sockfd; + const char *dst_host, *dst_port; + char *buf = NULL; + int buflen, offset = 0, rlen; + struct addrinfo hints, *result, *rp; + int i; + + dst_host = hddtemp_host ? hddtemp_host : DEFAULT_HOST; + dst_port = hddtemp_port ? hddtemp_port : DEFAULT_PORT; + + memset(&hints, 0, sizeof(hints)); + hints.ai_family = AF_INET; /* XXX: hddtemp has no ipv6 support (yet?) */ + hints.ai_socktype = SOCK_STREAM; + + if ((i = getaddrinfo(dst_host, dst_port, &hints, &result))) { + NORM_ERR("getaddrinfo(): %s", gai_strerror(i)); + return NULL; } - if (ret == 3) { - *port = n; - } else { - *port = PORT; + for (rp = result; rp; rp = rp->ai_next) { + sockfd = socket(rp->ai_family, rp->ai_socktype, rp->ai_protocol); + if (sockfd == -1) + continue; + if (connect(sockfd, rp->ai_addr, rp->ai_addrlen) != -1) + break; + close(sockfd); + } + if (!rp) { + NORM_ERR("could not connect to mpd host"); + goto GET_OUT; } - return 0; + buflen = 1024; + buf = malloc(buflen); + while ((rlen = recv(sockfd, buf + offset, buflen - offset - 1, 0)) > 0) { + offset += rlen; + if (buflen - offset < 1) { + buflen += 1024; + buf = realloc(buf, buflen); + } + } + if (rlen < 0) + perror("recv"); + + buf[offset] = '\0'; + + close(sockfd); +GET_OUT: + freeaddrinfo(result); + return buf; } /* this is an iterator: * set line to NULL in consecutive calls to get the next field - * returns "" or NULL on error + * note that exhausing iteration is assumed - otherwise *saveptr + * is not being freed! */ -static char *read_hdd_val(const char *line) +static int read_hdd_val(const char *line, char **dev, short *val, char *unit, + char **saveptr) { - static char line_s[512] = "\0"; + char *line_s, *cval, *endptr; static char *p = 0; - char *dev, *val, unit; - char *ret = NULL; if (line) { - if (!snprintf(line_s, 512, "%s", line)) return ret; - p = line_s; + *saveptr = strdup(line); + p = *saveptr; } - if (!(*line_s) || !p) - return ret; + line_s = *saveptr; + /* read the device */ - dev = ++p; + *dev = ++p; if (!(p = strchr(p, line_s[0]))) - return ret; + goto out_fail; *(p++) = '\0'; /* jump over the devname */ if (!(p = strchr(p, line_s[0]))) - return ret; + goto out_fail; /* read the value */ - val = ++p; + cval = ++p; if (!(p = strchr(p, line_s[0]))) - return ret; + goto out_fail; *(p++) = '\0'; - unit = *(p++); + *unit = *(p++); + *val = strtol(cval, &endptr, 10); + if (*endptr) + goto out_fail; + /* preset p for next call */ - p = strchr(p + 1, line_s[0]); + p++; - if (dev && *dev && val && *val) { - asprintf(&ret, "%s%c%s", dev, unit, val); - } - return ret; + return 0; +out_fail: + free(*saveptr); + return 1; } -/* returns or NULL on error or N/A */ -char *get_hddtemp_info(char *dev, char *hostaddr, int port) +void update_hddtemp(void) { + char *data, *dev, unit, *saveptr; + short val; + static double last_hddtemp_update = 0.0; + + /* limit tcp connection overhead */ + if (current_update_time - last_hddtemp_update < 5) + return; + last_hddtemp_update = current_update_time; + + free_hddtemp_info(); + + if (!(data = fetch_hddtemp_output())) + return; + + if (read_hdd_val(data, &dev, &val, &unit, &saveptr)) { + free(data); + return; + } + do { + add_hddtemp_info(dev, val, unit); + } while (!read_hdd_val(NULL, &dev, &val, &unit, &saveptr)); + free(data); +} + +void free_hddtemp(void) { - int sockfd = 0; - struct hostent he, *he_res = 0; - int he_errno; - char hostbuff[2048]; - struct sockaddr_in addr; - struct timeval tv; - fd_set rfds; - int len, i; - char *p, *r = NULL; - - if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) == -1) { - perror("socket"); - goto GET_OUT; + free_hddtemp_info(); + if (hddtemp_host) { + free(hddtemp_host); + hddtemp_host = NULL; } - -#ifdef HAVE_GETHOSTBYNAME_R - if (gethostbyname_r(hostaddr, &he, hostbuff, - sizeof(hostbuff), &he_res, &he_errno)) { - NORM_ERR("hddtemp gethostbyname_r: %s", hstrerror(h_errno)); -#else /* HAVE_GETHOSTBYNAME_R */ - if (!(he_res = gethostbyname(hostaddr))) { - perror("gethostbyname()"); -#endif /* HAVE_GETHOSTBYNAME_R */ - goto GET_OUT; + if (hddtemp_port) { + free(hddtemp_port); + hddtemp_port = NULL; } - - addr.sin_family = AF_INET; - addr.sin_port = htons(port); - addr.sin_addr = *((struct in_addr *) he_res->h_addr); - memset(&(addr.sin_zero), 0, 8); - - if (connect(sockfd, (struct sockaddr *) &addr, - sizeof(struct sockaddr)) == -1) { - perror("connect"); - goto GET_OUT; - } - - FD_ZERO(&rfds); - FD_SET(sockfd, &rfds); - - /* We're going to wait up to a half second to see whether there's any - * data available. Polling with timeout set to 0 doesn't seem to work - * with hddtemp. - */ - tv.tv_sec = 0; - tv.tv_usec = 500000; - - i = select(sockfd + 1, &rfds, NULL, NULL, &tv); - if (i == -1) { /* select() failed */ - if (errno == EINTR) { - /* silently ignore interrupted system call */ - goto GET_OUT; - } else { - perror("select"); - } - } else if (i == 0) { /* select() timeouted */ - NORM_ERR("hddtemp had nothing for us"); - goto GET_OUT; - } - - p = buf; - len = 0; - do { - i = recv(sockfd, p, BUFLEN - (p - buf), 0); - if (i < 0) { - perror("recv"); - break; - } - len += i; - p += i; - } while (i > 0 && p < buf + BUFLEN - 1); - - if (len < 2) { - NORM_ERR("hddtemp returned nada"); - goto GET_OUT; - } - - buf[len] = 0; - - if ((p = read_hdd_val(buf)) == NULL) - goto GET_OUT; - do { - if (!strncmp(dev, p, strlen(dev))) - asprintf(&r, "%s", p + strlen(dev)); - free(p); - } while(!r && (p = read_hdd_val(NULL)) != NULL); - -GET_OUT: - close(sockfd); - return r; +} + +int get_hddtemp_info(const char *dev, short *val, char *unit) +{ + struct hdd_info *hdi = hdd_info_head.next; + + /* if no dev is given, just use hdd_info_head->next */ + while(dev && hdi) { + if (!strcmp(dev, hdi->dev)) + break; + hdi = hdi->next; + } + if (!hdi) + return 1; + + *val = hdi->temp; + *unit = hdi->unit; + return 0; } diff --git a/src/hddtemp.h b/src/hddtemp.h index bfc897cf..1642e412 100644 --- a/src/hddtemp.h +++ b/src/hddtemp.h @@ -3,7 +3,10 @@ #ifndef HDDTEMP_H_ #define HDDTEMP_H_ -int scan_hddtemp(const char *arg, char **dev, char **addr, int *port); -char *get_hddtemp_info(char *dev, char *addr, int port); +void set_hddtemp_host(const char *); +void set_hddtemp_port(const char *); +void update_hddtemp(void); +void free_hddtemp(void); +int get_hddtemp_info(const char *, short *, char *); #endif /*HDDTEMP_H_*/