1
0
Fork 0
mirror of https://github.com/otakuto/crazydiskinfo.git synced 2025-07-21 10:39:52 +00:00
crazydiskinfo/main.cpp
2020-02-06 07:22:57 +09:00

571 lines
15 KiB
C++

#include <iostream>
#include <array>
#include <vector>
#include <algorithm>
#include <functional>
#include <locale.h>
#include <ncurses.h>
#include <atasmart.h>
#include <sys/signal.h>
#include <dirent.h>
std::string const TITLE = "CrazyDiskInfo";
std::string const VERSION = "1.1.0";
constexpr int const STATUS_WIDTH = 83;
constexpr int const DEVICE_BAR_HEIGHT = 4;
constexpr int const VERSION_HEIGHT = 1;
constexpr int const HEALTH_COLOR = 1;
constexpr int const HEALTH_INV_COLOR = 4;
constexpr int const ATTRIBUTE_LEGEND_COLOR = 7;
constexpr int const TITLE_COLOR = 8;
int width;
int height;
class Option
{
public:
bool hideSerial;
bool isRawHex;
Option()
:
hideSerial(false),
isRawHex(true)
{
}
};
enum class Health
{
Good,
Caution,
Bad
};
class Attribute
{
public:
uint8_t id;
std::string name;
uint8_t current;
uint8_t worst;
uint8_t threshold;
uint64_t raw;
};
class SMART
{
public:
std::string deviceName;
std::string model;
std::string firmware;
std::string serial;
//TODO
//use std::optional
std::pair<bool, uint64_t> capacity;
std::pair<bool, uint64_t> powerOnCount;
std::pair<bool, uint64_t> powerOnHour;
std::pair<bool, double> temperature;
std::vector<Attribute> attribute;
SMART(std::string deviceName)
:
deviceName(deviceName)
{
SkDisk * skdisk;
sk_disk_open(deviceName.c_str(), &skdisk);
sk_disk_smart_read_data(skdisk);
const SkIdentifyParsedData * data;
sk_disk_identify_parse(skdisk, &data);
model = data->model;
firmware = data->firmware;
serial = data->serial;
uint64_t value;
if (!sk_disk_get_size(skdisk, &value))
{
std::get<0>(capacity) = true;
std::get<1>(capacity) = value;
}
else
{
std::get<0>(capacity) = false;
}
if (!sk_disk_smart_get_power_cycle(skdisk, &value))
{
std::get<0>(powerOnCount) = true;
std::get<1>(powerOnCount) = value;
}
else
{
std::get<0>(powerOnCount) = false;
}
if (!sk_disk_smart_get_power_on(skdisk, &value))
{
std::get<0>(powerOnHour) = true;
std::get<1>(powerOnHour) = value / (1000llu * 60llu * 60llu);
}
else
{
std::get<0>(powerOnHour) = false;
}
if (!sk_disk_smart_get_temperature(skdisk, &value))
{
std::get<0>(temperature) = true;
std::get<1>(temperature) = (double)(value - 273150llu) / 1000.0;
}
else
{
std::get<0>(temperature) = false;
}
sk_disk_smart_parse_attributes(skdisk, [](SkDisk * skdisk, SkSmartAttributeParsedData const * data, void * userdata)
{
auto attribute = reinterpret_cast<std::vector<Attribute> *>(userdata);
Attribute attr = {};
attr.id = data->id;
attr.name = data->name;
attr.current = data->current_value;
attr.worst = data->worst_value;
attr.threshold = data->threshold;
for (int i = 0; i < 6; ++i)
{
attr.raw += data->raw[i] << (8 * i);
}
attribute->push_back(attr);
}, &attribute);
sk_disk_free(skdisk);
}
};
Health temperatureToHealth(double temperature)
{
if (temperature < 50)
{
return Health::Good;
}
else if (temperature < 55)
{
return Health::Caution;
}
else
{
return Health::Bad;
}
}
Health attributeToHealth(Attribute const & attribute)
{
if ((attribute.threshold != 0) && (attribute.current < attribute.threshold))
{
return Health::Bad;
}
else if (((attribute.id == 0x05) || (attribute.id == 0xC5) || (attribute.id == 0xC6)) && (attribute.raw != 0))
{
return Health::Caution;
}
else
{
return Health::Good;
}
}
Health smartToHealth(SMART const & smart)
{
return attributeToHealth(*std::max_element(smart.attribute.cbegin(), smart.attribute.cend(), [](Attribute const & lhs, Attribute const & rhs)
{
return static_cast<int>(attributeToHealth(lhs)) < static_cast<int>(attributeToHealth(rhs));
}));
}
std::string healthToString(Health health)
{
switch (health)
{
case Health::Good:
return "Good";
case Health::Caution:
return "Caution";
case Health::Bad:
return "Bad";
default:
return "Unknown";
}
}
void drawVersion(WINDOW * window)
{
wresize(window, VERSION_HEIGHT, width);
wattrset(window, COLOR_PAIR(HEALTH_COLOR));
mvwhline(window, 0, 0, '-', width);
wattroff(window, COLOR_PAIR(HEALTH_COLOR));
auto title = " " + TITLE + "-" + VERSION + " ";
wattrset(window, COLOR_PAIR(TITLE_COLOR));
mvwprintw(window, 0, (width - title.length()) / 2, title.c_str());
wattroff(window, COLOR_PAIR(TITLE_COLOR));
wnoutrefresh(window);
}
void drawDeviceBar(WINDOW * window, std::vector<SMART> const & smartList, int select)
{
int x = 0;
for (int i = 0; i < static_cast<int>(smartList.size()); ++i)
{
wattrset(window, COLOR_PAIR(HEALTH_INV_COLOR + static_cast<int>(smartToHealth(smartList[i]))));
mvwprintw(window, 0, x, "%-7s", healthToString(smartToHealth(smartList[i])).c_str());
wattroff(window, COLOR_PAIR(HEALTH_INV_COLOR + static_cast<int>(smartToHealth(smartList[i]))));
if (std::get<0>(smartList[i].temperature))
{
wattrset(window, COLOR_PAIR(HEALTH_INV_COLOR + static_cast<int>(temperatureToHealth(std::get<1>(smartList[i].temperature)))));
mvwprintw(window, 1, x, "%.1f ", smartList[i].temperature);
waddch(window, ACS_DEGREE);
waddstr(window, "C");
wattroff(window, COLOR_PAIR(HEALTH_INV_COLOR + static_cast<int>(temperatureToHealth(std::get<1>(smartList[i].temperature)))));
}
else
{
mvwprintw(window, 1, x, "-- ");
waddch(window, ACS_DEGREE);
waddstr(window, "C");
}
if (i == select)
{
wattrset(window, COLOR_PAIR(HEALTH_COLOR) | A_BOLD);
mvwprintw(window, 2, x, smartList[i].deviceName.c_str());
wattroff(window, COLOR_PAIR(HEALTH_COLOR) | A_BOLD);
wattrset(window, COLOR_PAIR(HEALTH_COLOR));
mvwhline(window, 3, x, '-', smartList[i].deviceName.length());
wattroff(window, COLOR_PAIR(HEALTH_COLOR));
}
else
{
mvwprintw(window, 2, x, smartList[i].deviceName.c_str());
mvwhline(window, 3, x, ' ', smartList[i].deviceName.length());
}
x += smartList[i].deviceName.length() + 1;
}
pnoutrefresh(window, 0, 0, 1, 0, DEVICE_BAR_HEIGHT, width - 1);
}
void drawStatus(WINDOW * window, SMART const & smart, Option const & option)
{
wresize(window, 10 + smart.attribute.size(), STATUS_WIDTH);
wborder(window, '|', '|', '-', '-', '+', '+', '+', '+');
if (std::get<0>(smart.capacity))
{
std::vector<std::string> unit = {{"Byte", "KiB", "MiB", "GiB", "TiB", "PiB", "EiB", "ZiB", "YiB"}};
int u = 0;
double capacity = std::get<1>(smart.capacity);
while (true)
{
double old = capacity;
capacity /= 1024;
if (capacity < 1.0)
{
capacity = old;
break;
}
++u;
}
char s[STATUS_WIDTH];
int len = snprintf(s, STATUS_WIDTH, " %s [%.1f %s] ", smart.model.c_str(), capacity, unit[u].c_str());
wattrset(window, COLOR_PAIR(HEALTH_COLOR) | A_BOLD);
mvwprintw(window, 0, (STATUS_WIDTH - len) / 2, "%s", s);
wattroff(window, COLOR_PAIR(HEALTH_COLOR) | A_BOLD);
}
else
{
char s[STATUS_WIDTH];
int len = snprintf(s, STATUS_WIDTH, " %s [--] ", smart.model.c_str());
wattrset(window, COLOR_PAIR(HEALTH_COLOR) | A_BOLD);
mvwprintw(window, 0, (STATUS_WIDTH - len) / 2, "%s", s);
wattroff(window, COLOR_PAIR(HEALTH_COLOR) | A_BOLD);
}
wattrset(window, COLOR_PAIR(HEALTH_COLOR));
mvwprintw(window, 2, (int)(STATUS_WIDTH * (1.0 / 5)), "Firmware:");
wattroff(window, COLOR_PAIR(HEALTH_COLOR));
wattrset(window, COLOR_PAIR(HEALTH_COLOR) | A_BOLD);
wprintw(window, " %s", smart.firmware.c_str());
wattroff(window, COLOR_PAIR(HEALTH_COLOR) | A_BOLD);
wattrset(window, COLOR_PAIR(HEALTH_COLOR));
mvwprintw(window, 3, (int)(STATUS_WIDTH * (1.0 / 5)), "Serial: ");
wattroff(window, COLOR_PAIR(HEALTH_COLOR));
wattrset(window, COLOR_PAIR(HEALTH_COLOR) | A_BOLD);
if (option.hideSerial)
{
wprintw(window, " ********************");
}
else
{
wprintw(window, " %s", smart.serial.c_str());
}
wattroff(window, COLOR_PAIR(HEALTH_COLOR) | A_BOLD);
wattrset(window, COLOR_PAIR(HEALTH_COLOR));
mvwprintw(window, 1, 1, "Status");
wattroff(window, COLOR_PAIR(HEALTH_COLOR));
wattrset(window, COLOR_PAIR(HEALTH_INV_COLOR + static_cast<int>(smartToHealth(smart))));
mvwprintw(window, 2, 2, "+--------+");
mvwprintw(window, 3, 2, "| |");
mvwprintw(window, 4, 2, "+--------+");
mvwprintw(window, 3, 2 + ((sizeof("| |") - healthToString(smartToHealth(smart)).length()) / 2), "%s", healthToString(smartToHealth(smart)).c_str());
wattroff(window, COLOR_PAIR(HEALTH_INV_COLOR + static_cast<int>(smartToHealth(smart))));
if (std::get<0>(smart.temperature))
{
wattrset(window, COLOR_PAIR(HEALTH_COLOR));
mvwprintw(window, 5, 1, "Temperature");
wattroff(window, COLOR_PAIR(HEALTH_COLOR));
wattrset(window, COLOR_PAIR(HEALTH_INV_COLOR + static_cast<int>(temperatureToHealth(std::get<1>(smart.temperature)))));
mvwprintw(window, 6, 2, " %0.1f ", std::get<1>(smart.temperature));
waddch(window, ACS_DEGREE);
waddstr(window, "C ");
wattroff(window, COLOR_PAIR(HEALTH_INV_COLOR + static_cast<int>(temperatureToHealth(std::get<1>(smart.temperature)))));
}
else
{
mvwprintw(window, 5, 1, "Temperature");
mvwprintw(window, 6, 2, " -- ");
waddch(window, ACS_DEGREE);
waddstr(window, "C ");
}
if (std::get<0>(smart.powerOnCount))
{
wattrset(window, COLOR_PAIR(HEALTH_COLOR));
mvwprintw(window, 2, (int)(STATUS_WIDTH * (3.0 / 5)), "Power On Count:");
wattroff(window, COLOR_PAIR(HEALTH_COLOR));
wattrset(window, COLOR_PAIR(HEALTH_COLOR) | A_BOLD);
wprintw(window, " %llu ", std::get<1>(smart.powerOnCount));
wattroff(window, COLOR_PAIR(HEALTH_COLOR) | A_BOLD);
wattrset(window, COLOR_PAIR(HEALTH_COLOR));
wprintw(window, "count");
}
else
{
mvwprintw(window, 2, (int)(STATUS_WIDTH * (3.0 / 5)), "Power On Count:");
wprintw(window, " -- count");
}
if (std::get<0>(smart.powerOnHour))
{
wattrset(window, COLOR_PAIR(HEALTH_COLOR));
mvwprintw(window, 3, (int)(STATUS_WIDTH * (3.0 / 5)), "Power On Hours:");
wattroff(window, COLOR_PAIR(HEALTH_COLOR));
wattrset(window, COLOR_PAIR(HEALTH_COLOR) | A_BOLD);
wprintw(window, " %llu ", std::get<1>(smart.powerOnHour));
wattroff(window, COLOR_PAIR(HEALTH_COLOR) | A_BOLD);
wattrset(window, COLOR_PAIR(HEALTH_COLOR));
wprintw(window, "hours");
wattroff(window, COLOR_PAIR(HEALTH_COLOR));
}
else
{
mvwprintw(window, 3, (int)(STATUS_WIDTH * (3.0 / 5)), "Power On Hours:");
wprintw(window, " -- hours");
}
wattrset(window, COLOR_PAIR(ATTRIBUTE_LEGEND_COLOR));
mvwprintw(window, 8, 1, " Status ID AttributeName Current Worst Threshold RawValues(%s) ", option.isRawHex ? "Hex" : "Dec");
wattroff(window, COLOR_PAIR(ATTRIBUTE_LEGEND_COLOR));
auto attributeFormat = option.isRawHex ? " %-7s %02X %-28s %7d %5d %9d %012llX " : " %-7s %02X %-28s %7d %5d %9d %15llu ";
for (int i = 0; i < static_cast<int>(smart.attribute.size()); ++i)
{
wattrset(window, COLOR_PAIR(HEALTH_COLOR + static_cast<int>(attributeToHealth(smart.attribute[i]))));
mvwprintw(window, 9 + i, 1, attributeFormat,
healthToString(attributeToHealth(smart.attribute[i])).c_str(),
smart.attribute[i].id,
smart.attribute[i].name.c_str(),
smart.attribute[i].current,
smart.attribute[i].worst,
smart.attribute[i].threshold,
smart.attribute[i].raw);
wattroff(window, COLOR_PAIR(HEALTH_COLOR + static_cast<int>(attributeToHealth(smart.attribute[i]))));
}
pnoutrefresh(window, 0, 0,
5, std::max(0, (width - STATUS_WIDTH) / 2),
std::min(height - 1, 5 + 10 + static_cast<int>(smart.attribute.size())), std::min(width - 1, std::max(0, (width - STATUS_WIDTH) / 2) + STATUS_WIDTH));
}
std::function<void(void)> update;
void actionWINCH(int)
{
clear();
endwin();
refresh();
update();
}
int main()
{
setlocale(LC_ALL, "");
initscr();
cbreak();
noecho();
curs_set(0);
getmaxyx(stdscr, height, width);
start_color();
init_pair(HEALTH_COLOR, COLOR_CYAN, COLOR_BLACK);
init_pair(HEALTH_COLOR + 1, COLOR_BLACK, COLOR_YELLOW);
init_pair(HEALTH_COLOR + 2, COLOR_WHITE, COLOR_RED);
init_pair(HEALTH_INV_COLOR, COLOR_BLACK, COLOR_CYAN);
init_pair(HEALTH_INV_COLOR + 1, COLOR_BLACK, COLOR_YELLOW);
init_pair(HEALTH_INV_COLOR + 2, COLOR_WHITE, COLOR_RED);
init_pair(ATTRIBUTE_LEGEND_COLOR, COLOR_BLACK, COLOR_GREEN);
init_pair(TITLE_COLOR, COLOR_YELLOW, COLOR_BLACK);
std::vector<SMART> smartList;
auto dir = opendir("/sys/block");
while (auto e = readdir(dir))
{
if (std::string(".") != std::string(e->d_name) &&
std::string("..") != std::string(e->d_name) &&
std::string("ram") != std::string(e->d_name).substr(0,3) &&
std::string("loop") != std::string(e->d_name).substr(0,4))
{
SkDisk * skdisk;
SkBool b;
int f = sk_disk_open((std::string("/dev/") + std::string(e->d_name)).c_str(), &skdisk);
if (f < 0)
{
continue;
}
int smart_ret = sk_disk_smart_is_available(skdisk, &b);
sk_disk_free(skdisk);
if (smart_ret < 0)
{
continue;
}
if (b)
{
smartList.push_back(SMART(std::string("/dev/") + std::string(e->d_name)));
}
}
}
std::sort(smartList.begin(), smartList.end(), [](SMART const & lhs, SMART const & rhs){return lhs.deviceName < rhs.deviceName;});
if (smartList.size() == 0)
{
endwin();
std::cerr << "No S.M.A.R.T readable devices." << std::endl;
std::cerr << "If you are non-root user, please use sudo or become root." << std::endl;
return 1;
}
int select = 0;
Option option;
WINDOW * windowVersion;
windowVersion = newwin(1, width, 0, 0);
WINDOW * windowDeviceBar;
{
int x = 0;
for (auto && e : smartList)
{
x += e.deviceName.length() + 1;
}
windowDeviceBar = newpad(DEVICE_BAR_HEIGHT, x);
keypad(windowDeviceBar, true);
}
WINDOW * windowDeviceStatus;
windowDeviceStatus = newpad(10 + smartList[select].attribute.size(), STATUS_WIDTH);
update = [&]()
{
getmaxyx(stdscr, height, width);
wclear(windowVersion);
drawVersion(windowVersion);
wclear(windowDeviceBar);
drawDeviceBar(windowDeviceBar, smartList, select);
wclear(windowDeviceStatus);
drawStatus(windowDeviceStatus, smartList[select], option);
doupdate();
};
update();
{
struct sigaction s = {{actionWINCH}};
sigaction(SIGWINCH, &s, nullptr);
}
while (true)
{
switch (wgetch(windowDeviceBar))
{
case KEY_HOME:
select = 0;
clear();
refresh();
update();
break;
case KEY_END:
select = static_cast<int>(smartList.size()) - 1;
clear();
refresh();
update();
break;
case KEY_LEFT:
case 'h':
select = std::max(select - 1, 0);
clear();
refresh();
update();
break;
case KEY_RIGHT:
case 'l':
select = std::min(select + 1, static_cast<int>(smartList.size()) - 1);
clear();
refresh();
update();
break;
case 's':
option.hideSerial = !option.hideSerial;
clear();
refresh();
update();
break;
case 'd':
option.isRawHex = !option.isRawHex;
clear();
refresh();
update();
break;
case 'q':
endwin();
return 0;
default:
break;
}
}
}