X-Git-Url: https://git.wpitchoune.net/gitweb/?p=psensor.git;a=blobdiff_plain;f=src%2Flib%2Fhdd_atasmart.c;h=402c92ee949ce9c6ea24a93ed32e37b81bb88191;hp=592fcbd60a3d72635cca01ad658a7c0f8c2d7978;hb=c1e20f2631a1249720e9c75d753eacfcb0f6c7b9;hpb=12a0e6ab2388b9e7a67dd0582c45fb3305836c09 diff --git a/src/lib/hdd_atasmart.c b/src/lib/hdd_atasmart.c index 592fcbd..402c92e 100644 --- a/src/lib/hdd_atasmart.c +++ b/src/lib/hdd_atasmart.c @@ -1,5 +1,5 @@ /* - * Copyright (C) 2010-2011 jeanfi@gmail.com + * Copyright (C) 2010-2014 jeanfi@gmail.com * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as @@ -16,25 +16,186 @@ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA * 02110-1301 USA */ +#define _LARGEFILE_SOURCE 1 +#include "config.h" + #include #include #define _(str) gettext(str) +#include +#include +#include #include +#include +#include +#include +#include +#include + #include +#include + +#include +#include +#include -#include "hdd.h" -#include "log.h" +static const char *PROVIDER_NAME = "atasmart"; -struct psensor **hdd_psensor_list_add(struct psensor **sensors, - int values_max_length) +static int filter_sd(const char *p) { + return strlen(p) == 8 && !strncmp(p, "/dev/sd", 7); +} - log_debug("hdd_psensor_list_add"); +static void provider_data_free(void *data) +{ + sk_disk_free((SkDisk *)data); +} - return sensors; +static SkDisk *get_disk(struct psensor *s) +{ + return (SkDisk *)s->provider_data; } -void hdd_psensor_list_update(struct psensor **sensors) +static struct psensor * +create_sensor(char *id, char *name, SkDisk *disk, int values_max_length) { + struct psensor *s; + int t; + + t = SENSOR_TYPE_ATASMART | SENSOR_TYPE_HDD | SENSOR_TYPE_TEMP; + + s = psensor_create(id, + strdup(name), + strdup(_("Disk")), + t, + values_max_length); + + s->provider_data = disk; + s->provider_data_free_fct = &provider_data_free; + + return s; +} + +/* + * Performs the same tests than sk_disk_open and outputs the result. + */ +static void analyze_disk(const char *dname) +{ + int f; + struct stat st; + uint64_t size; + + log_fct("Analyze %s", dname); + + f = open(dname, O_RDONLY|O_NOCTTY|O_NONBLOCK|O_CLOEXEC); + + if (f < 0) { + log_fct("Could not open file %s: %s", dname, strerror(errno)); + goto fail; + } + + if (fstat(f, &st) < 0) { + log_fct("fstat fails %s: %s", dname, strerror(errno)); + goto fail; + } + + if (!S_ISBLK(st.st_mode)) { + log_fct("!S_ISBLK fails %s", dname); + goto fail; + } + + size = (uint64_t)-1; + /* So, it's a block device. Let's make sure the ioctls work */ + if (ioctl(f, BLKGETSIZE64, &size) < 0) { + log_fct("ioctl fails %s: %s", dname, strerror(errno)); + goto fail; + } + + if (size <= 0 || size == (uint64_t) -1) { + log_fct("ioctl wrong size %s: %ld", dname, size); + goto fail; + } + + fail: + close(f); +} + +void +atasmart_psensor_list_append(struct psensor ***sensors, int values_max_length) +{ + char **paths, **tmp, *id; + SkDisk *disk; + struct psensor *sensor; + + log_fct_enter(); + + paths = dir_list("/dev", filter_sd); + + tmp = paths; + while (*tmp) { + log_fct("Open %s", *tmp); + + if (!sk_disk_open(*tmp, &disk)) { + id = malloc(strlen(PROVIDER_NAME) + + 1 + + strlen(*tmp) + + 1); + sprintf(id, "%s %s", PROVIDER_NAME, *tmp); + + sensor = create_sensor(id, + *tmp, + disk, + values_max_length); + + psensor_list_append(sensors, sensor); + } else { + log_err(_("%s: sk_disk_open() failure: %s."), + PROVIDER_NAME, + *tmp); + analyze_disk(*tmp); + } + + tmp++; + } + + paths_free(paths); + + log_fct_exit(); +} + +void atasmart_psensor_list_update(struct psensor **sensors) +{ + struct psensor **cur, *s; + uint64_t kelvin; + int ret; + double c; + SkDisk *disk; + + if (!sensors) + return; + + cur = sensors; + while (*cur) { + s = *cur; + if (!(s->type & SENSOR_TYPE_REMOTE) + && s->type & SENSOR_TYPE_ATASMART) { + disk = get_disk(s); + + ret = sk_disk_smart_read_data(disk); + + if (!ret) { + ret = sk_disk_smart_get_temperature(disk, + &kelvin); + + if (!ret) { + c = (kelvin - 273150) / 1000; + psensor_set_current_value(s, c); + log_fct("%s %.2f", s->id, c); + } + } + } + + cur++; + } }