1
0
mirror of https://github.com/Llewellynvdm/conky.git synced 2025-01-13 19:22:58 +00:00

Solaris support (#445)

* Basic Solaris support

* Fixed indentation & minor improvements

* support for diskio
This commit is contained in:
Jan Senolt 2018-01-19 15:00:42 +01:00 committed by Brenden Matthews
parent 262a2925bc
commit b814aafc6e
10 changed files with 538 additions and 32 deletions

View File

@ -39,17 +39,19 @@ if(CMAKE_SYSTEM_NAME MATCHES "OpenBSD")
set(OS_OPENBSD true)
endif(CMAKE_SYSTEM_NAME MATCHES "OpenBSD")
if(CMAKE_SYSTEM_NAME MATCHES "Solaris")
if(CMAKE_SYSTEM_NAME MATCHES "SunOS")
set(OS_SOLARIS true)
endif(CMAKE_SYSTEM_NAME MATCHES "Solaris")
endif(CMAKE_SYSTEM_NAME MATCHES "SunOS")
if(CMAKE_SYSTEM_NAME MATCHES "NetBSD")
set(OS_NETBSD true)
endif(CMAKE_SYSTEM_NAME MATCHES "NetBSD")
if(NOT OS_LINUX AND NOT OS_FREEBSD AND NOT OS_OPENBSD AND NOT OS_DRAGONFLY)
if(NOT OS_LINUX AND NOT OS_FREEBSD AND NOT OS_OPENBSD AND NOT OS_DRAGONFLY
AND NOT OS_SOLARIS)
message(FATAL_ERROR "Your platform, '${CMAKE_SYSTEM_NAME}', is not currently supported. Patches are welcome.")
endif(NOT OS_LINUX AND NOT OS_FREEBSD AND NOT OS_OPENBSD AND NOT OS_DRAGONFLY)
endif(NOT OS_LINUX AND NOT OS_FREEBSD AND NOT OS_OPENBSD AND NOT OS_DRAGONFLY
AND NOT OS_SOLARIS)
include(FindThreads)
find_package(Threads)
@ -66,6 +68,10 @@ set(conky_libs ${conky_libs} -L/usr/pkg/lib)
set(conky_includes ${conky_includes} -I/usr/pkg/include)
endif(OS_DRAGONFLY)
if(OS_SOLARIS)
set(conky_libs ${conky_libs} -L/usr/local/lib)
endif(OS_SOLARIS)
# Do version stuff
set(VERSION_MAJOR "1")
set(VERSION_MINOR "10")

View File

@ -66,17 +66,20 @@ if(CMAKE_SYSTEM_NAME MATCHES "OpenBSD")
set(OS_OPENBSD true)
endif(CMAKE_SYSTEM_NAME MATCHES "OpenBSD")
if(CMAKE_SYSTEM_NAME MATCHES "Solaris")
if(CMAKE_SYSTEM_NAME MATCHES "SunOS")
set(OS_SOLARIS true)
endif(CMAKE_SYSTEM_NAME MATCHES "Solaris")
set(conky_libs ${conky_libs} -lkstat)
endif(CMAKE_SYSTEM_NAME MATCHES "SunOS")
if(CMAKE_SYSTEM_NAME MATCHES "NetBSD")
set(OS_NETBSD true)
endif(CMAKE_SYSTEM_NAME MATCHES "NetBSD")
if(NOT OS_LINUX AND NOT OS_FREEBSD AND NOT OS_OPENBSD AND NOT OS_DRAGONFLY)
if(NOT OS_LINUX AND NOT OS_FREEBSD AND NOT OS_OPENBSD AND NOT OS_DRAGONFLY
AND NOT OS_SOLARIS)
message(FATAL_ERROR "Your platform, '${CMAKE_SYSTEM_NAME}', is not currently supported. Patches are welcome.")
endif(NOT OS_LINUX AND NOT OS_FREEBSD AND NOT OS_OPENBSD AND NOT OS_DRAGONFLY)
endif(NOT OS_LINUX AND NOT OS_FREEBSD AND NOT OS_OPENBSD AND NOT OS_DRAGONFLY
AND NOT OS_SOLARIS)
if(BUILD_I18N AND OS_DRAGONFLY)
set(conky_libs ${conky_libs} -lintl)

View File

@ -252,6 +252,20 @@ enum {
#define KFLAG_FLIP(a) info.kflags ^= a
#define KFLAG_ISSET(a) info.kflags & a
#if !defined(MAX)
#define MAX(x,y) \
({ __typeof__ (x) _x = (x); \
__typeof__ (y) _y = (y); \
_x > _y ? _x : _y;})
#endif
#if !defined(MIN)
#define MIN(x,y) \
({ __typeof__ (x) _x = (x); \
__typeof__ (y) _y = (y); \
_x < _y ? _x : _y;})
#endif
/* defined in conky.c, needed by top.c */
extern int top_cpu, top_mem, top_time;
#ifdef BUILD_IOSTATS

View File

@ -87,11 +87,17 @@ struct diskio_stat *prepare_diskio_stat(const char *s)
#endif
strncpy(&(device_name[0]), &device_s[0], text_buffer_size.get(*state));
#if !defined(__sun)
/*
* On Solaris we currently don't use the name of disk's special file so
* this test is useless.
*/
snprintf(&(stat_name[0]), text_buffer_size.get(*state), "/dev/%s", &(device_name[0]));
if (stat(&(stat_name[0]), &sb) || !S_ISBLK(sb.st_mode)) {
NORM_ERR("diskio device '%s' does not exist", &device_s[0]);
}
#endif
/* lookup existing */
while (cur->next) {

View File

@ -41,6 +41,8 @@
#include "dragonfly.h"
#elif defined(__OpenBSD__)
#include "openbsd.h"
#elif defined(__sun)
#include "solaris.h"
#endif
struct _entropy {

View File

@ -43,6 +43,11 @@
#include <sys/statfs.h>
#endif
#if defined(__sun)
#include <sys/types.h>
#include <sys/statvfs.h>
#endif
#if defined(__FreeBSD__)
#include "freebsd.h"
#elif defined(__DragonFly__)
@ -51,7 +56,8 @@
#if !defined(HAVE_STRUCT_STATFS_F_FSTYPENAME) && \
!defined (__OpenBSD__) && !defined(__FreeBSD__) && !defined(__DragonFly__)
!defined (__OpenBSD__) && !defined(__FreeBSD__) && \
!defined(__DragonFly__) && !defined(__sun)
#include <mntent.h>
#endif
@ -117,6 +123,15 @@ struct fs_stat *prepare_fs_stat(const char *s)
static void update_fs_stat(struct fs_stat *fs)
{
#if defined(__sun)
struct statvfs s;
if (statvfs(fs->path, &s) == 0) {
fs->size = (long long)s.f_blocks * s.f_frsize;
fs->avail = (long long)s.f_bavail * s.f_frsize;
fs->free = (long long)s.f_bfree * s.f_frsize;
(void) strncpy(fs->type, s.f_basetype, sizeof (fs->type));
#else
struct statfs64 s;
if (statfs64(fs->path, &s) == 0) {
@ -125,6 +140,7 @@ static void update_fs_stat(struct fs_stat *fs)
fs->avail = (long long)s.f_bavail * s.f_bsize;
fs->free = (long long)s.f_bfree * s.f_bsize;
get_fs_type(fs->path, fs->type);
#endif
} else {
NORM_ERR("statfs64 '%s': %s", fs->path, strerror(errno));
fs->size = 0;
@ -147,7 +163,8 @@ void get_fs_type(const char *path, char *result)
NORM_ERR("statfs64 '%s': %s", path, strerror(errno));
}
return;
#elif defined(__sun)
assert(0); /* not used - see update_fs_stat() */
#else /* HAVE_STRUCT_STATFS_F_FSTYPENAME */
struct mntent *me;

View File

@ -48,6 +48,11 @@
#endif /* __OpenBSD__ */
#endif /* HAVE_LINUX_SOUNDCARD_H */
#if defined(__sun)
#include <unistd.h>
#include <stropts.h>
#endif
#define MIXER_DEV "/dev/mixer"
static int mixer_fd;

View File

@ -40,6 +40,9 @@
#include <string.h>
#include <sys/ioctl.h>
#include <unistd.h>
#if defined(__sun)
#include <sys/sockio.h>
#endif
/* network interface stuff */

View File

@ -27,27 +27,95 @@
*
*/
/* doesn't work, feel free to finish this */
#include "conky.h"
#include "common.h"
#include <kstat.h>
#include <unistd.h>
#include <limits.h>
#include <sys/loadavg.h>
#include <sys/stat.h>
#include <sys/swap.h>
#include <sys/task.h>
#include <sys/sysinfo.h>
#include <sys/socket.h>
#include <procfs.h>
#include <project.h>
#include <dirent.h>
#include <net/if.h>
#include <sys/sockio.h>
#include "diskio.h"
#include "top.h"
#include <assert.h>
#include "solaris.h"
#include "net_stat.h"
static kstat_ctl_t *kstat;
static int kstat_updated;
static time_t kstat_updated;
static int pageshift = INT_MAX;
static int pagetok(int pages)
{
if (pageshift == INT_MAX) {
int pagesize = sysconf(_SC_PAGESIZE);
pageshift = 0;
while ((pagesize >>= 1) > 0)
pageshift++;
pageshift -= 10; /* 2^10 = 1024 */
}
return (pageshift > 0 ? pages << pageshift : pages >> -pageshift);
}
static void update_kstat()
{
time_t now = time(NULL);
if (kstat == NULL) {
kstat = kstat_open();
if (kstat == NULL) {
if ((kstat = kstat_open()) == NULL) {
NORM_ERR("can't open kstat: %s", strerror(errno));
return;
}
kstat_updated = 0;
}
if (now - kstat_updated < 2) /* Do not update kstats too often */
return;
if (kstat_chain_update(kstat) == -1) {
perror("kstat_chain_update");
return;
}
kstat_updated = now;
}
static kstat_named_t *get_kstat(const char *module, int inst, const char *name,
const char *stat)
{
kstat_t *ksp;
update_kstat();
ksp = kstat_lookup(kstat, (char *)module, inst, (char *)name);
if (ksp == NULL) {
NORM_ERR("cannot lookup kstat %s:%d:%s:%s %s", module, inst, name,
stat, strerror(errno));
return NULL;
}
if (kstat_read(kstat, ksp, NULL) >= 0) {
if (ksp->ks_type == KSTAT_TYPE_NAMED ||
ksp->ks_type == KSTAT_TYPE_TIMER) {
kstat_named_t *knp = (kstat_named_t *)kstat_data_lookup(ksp,
(char *)stat);
return knp;
} else {
NORM_ERR("kstat %s:%d:%s:%s has unexpected type %d",
module, inst, name, stat, ksp->ks_type);
return NULL;
}
}
NORM_ERR("cannot read kstat %s:%d:%s:%s", module, inst, name, stat);
return NULL;
}
void prepare_update()
@ -55,33 +123,407 @@ void prepare_update()
kstat_updated = 0;
}
double get_uptime()
int update_meminfo()
{
kstat_t *ksp;
update_kstat();
ksp = kstat_lookup(kstat, "unix", -1, "system_misc");
if (ksp != NULL) {
if (kstat_read(kstat, ksp, NULL) >= 0) {
kstat_named_t *knp;
int nswap = swapctl(SC_GETNSWP, 0);
struct swaptable *swt;
struct swapent *swe;
char path[PATH_MAX];
unsigned long stp, sfp;
knp = (kstat_named_t *) kstat_data_lookup(ksp, "boot_time");
if (knp != NULL) {
return get_time() - (double) knp->value.ui32;
}
}
}
}
/* RAM stats */
knp = get_kstat("unix", -1, "system_pages", "freemem");
if (knp != NULL)
info.memfree = pagetok(knp->value.ui32);
info.memmax = pagetok(sysconf(_SC_PHYS_PAGES));
if (info.memmax > info.memfree)
info.mem = info.memmax - info.memfree;
else /* for non-global zones with capped memory */
info.mem = info.memmax;
void update_meminfo()
{
/* TODO */
/* Swap stats */
if (nswap < 1)
return 0;
/* for swapctl(2) */
swt = (struct swaptable *)malloc(nswap * sizeof (struct swapent) +
sizeof (int));
if (swt == NULL)
return 0;
swt->swt_n = nswap;
swe = &(swt->swt_ent[0]);
/* We are not interested in ste_path */
for (int i = 0; i < nswap; i++)
swe[i].ste_path = path;
nswap = swapctl(SC_LIST, swt);
swe = &(swt->swt_ent[0]);
stp = sfp = 0;
for (int i = 0; i < nswap; i++) {
if ((swe[i].ste_flags & ST_INDEL) || (swe[i].ste_flags & ST_DOINGDEL))
continue;
stp += swe->ste_pages;
sfp += swe->ste_free;
}
free(swt);
info.swapfree = pagetok(sfp);
info.swapmax = pagetok(stp);
info.swap = info.swapmax - info.swapfree;
return 0;
}
int check_mount(struct text_object *obj)
{
/* stub */
(void)obj;
(void) obj;
return 0;
}
double get_battery_perct_bar(struct text_object *obj)
{
/* Not implemented */
(void) obj;
return 100.0;
}
double get_acpi_temperature(int fd)
{
/* Not implemented */
(void) fd;
return 0.0;
}
int update_total_processes(void)
{
kstat_named_t *knp = get_kstat("unix", -1, "system_misc", "nproc");
if (knp != NULL)
info.procs = knp->value.ui32;
return 0;
}
void get_battery_stuff(char *buf, unsigned int n, const char *bat, int item)
{
/* Not implemented */
}
int update_running_processes(void)
{
/* There is no kstat for this, see update_proc_entry() */
return 0;
}
int update_net_stats(void)
{
struct ifconf ifc;
int sockfd;
char buf[1024];
double d = current_update_time - last_update_time;
if (d < 0.1)
return 0;
/* Find all active net interfaces */
if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
NORM_ERR("cannot create socket: %s", strerror(errno));
return 0;
}
ifc.ifc_buf = buf;
ifc.ifc_len = sizeof (buf);
if (ioctl(sockfd, SIOCGIFCONF, &ifc) < 0) {
NORM_ERR("ioctl(SIOCGIFCONF) failed: %s", strerror(errno));
(void) close(sockfd);
return 0;
}
(void) close(sockfd);
/* Collect stats for all active interfaces */
for (int i = 0; i < ifc.ifc_len / sizeof (struct ifreq); i++) {
struct net_stat *ns;
struct ifreq *ifr = &ifc.ifc_req[i];
long long last_recv, last_trans;
long long r, t;
kstat_named_t *knp;
ns = get_net_stat((const char *) ifr->ifr_name, NULL, NULL);
ns->up = 1;
memcpy(&(ns->addr), &ifr->ifr_addr, sizeof (ifr->ifr_addr));
/* Skip the loopback interface, it does not have kstat data */
if (ifr->ifr_flags & IFF_LOOPBACK || strcmp(ifr->ifr_name, "lo0") == 0)
continue;
last_recv = ns->recv;
last_trans = ns->trans;
/* Get received bytes */
knp = get_kstat("link", -1, ifr->ifr_name, "rbytes");
if (knp == NULL) {
NORM_ERR("cannot read rbytes for %s\n", ifr->ifr_name);
continue;
}
r = (long long)knp->value.ui32;
if (r <= ns->last_read_recv) {
ns->recv += ((long long) 4294967295U - ns->last_read_recv) + r;
} else {
ns->recv += (r - ns->last_read_recv);
}
ns->last_read_recv = r;
/* Get transceived bytes */
knp = get_kstat("link", -1, ifr->ifr_name, "obytes");
if (knp == NULL) {
NORM_ERR("cannot read obytes for %s\n", ifr->ifr_name);
continue;
}
t = (long long)knp->value.ui32;
if (t < ns->last_read_trans) {
ns->trans += ((long long) 4294967295U - ns->last_read_trans) + t;
} else {
ns->trans += (t - ns->last_read_trans);
}
ns->last_read_trans = t;
ns->recv_speed = (ns->recv - last_recv) / d;
ns->trans_speed = (ns->trans - last_trans) / d;
}
return 0;
}
int update_cpu_usage(void)
{
static int last_cpu_cnt = 0;
static int *last_cpu_use = NULL;
double d = current_update_time - last_update_time;
int cpu;
if (d < 0.1)
return 0;
update_kstat();
info.cpu_count = sysconf(_SC_NPROCESSORS_ONLN);
/* (Re)allocate the array with previous values */
if (last_cpu_cnt != info.cpu_count || last_cpu_use == NULL) {
last_cpu_use = (int *)realloc(last_cpu_use,
info.cpu_count * sizeof (int));
last_cpu_cnt = info.cpu_count;
if (last_cpu_use == NULL)
return 0;
}
info.cpu_usage = (float *)malloc(info.cpu_count * sizeof (float));
for (cpu = 0; cpu < info.cpu_count; cpu++) {
char stat_name[PATH_MAX];
unsigned long cpu_user, cpu_nice, cpu_system, cpu_idle;
unsigned long cpu_use;
cpu_stat_t *cs;
kstat_t *ksp;
snprintf(stat_name, PATH_MAX, "cpu_stat%d", cpu);
ksp = kstat_lookup(kstat, (char *)"cpu_stat", cpu, stat_name);
if (ksp == NULL)
continue;
if (kstat_read(kstat, ksp, NULL) == -1)
continue;
cs = (cpu_stat_t *)ksp->ks_data;
cpu_idle = cs->cpu_sysinfo.cpu[CPU_IDLE];
cpu_user = cs->cpu_sysinfo.cpu[CPU_USER];
cpu_nice = cs->cpu_sysinfo.cpu[CPU_WAIT];
cpu_system = cs->cpu_sysinfo.cpu[CPU_KERNEL];
cpu_use = cpu_user + cpu_nice + cpu_system;
info.cpu_usage[cpu] = (double)(cpu_use - last_cpu_use[cpu]) / d / 100.0;
last_cpu_use[cpu] = cpu_use;
}
return 0;
}
void update_proc_entry(struct process *p)
{
psinfo_t proc;
int fd;
char pfn[PATH_MAX];
snprintf(pfn, PATH_MAX, "/proc/%d/psinfo", p->pid);
/* Ignore errors here as the process can be gone */
if ((fd = open(pfn, O_RDONLY)) < 0)
return;
if (pread(fd, &proc, sizeof (psinfo_t), 0) != sizeof (psinfo_t)) {
(void) close(fd);
return;
}
(void) close(fd);
free_and_zero(p->name);
free_and_zero(p->basename);
p->name = strndup(proc.pr_fname, text_buffer_size.get(*::state));
p->basename = strndup(proc.pr_fname, text_buffer_size.get(*::state));
p->uid = proc.pr_uid;
/* see proc(4) */
p->amount = (double)proc.pr_pctcpu / (double)0x8000 * 100.0;
p->rss = proc.pr_rssize * 1024; /* to bytes */
p->vsize = proc.pr_size * 1024; /* to bytes */
p->total_cpu_time = proc.pr_time.tv_sec * 100; /* to hundredths of secs */
if (proc.pr_lwp.pr_sname == 'O' || proc.pr_lwp.pr_sname == 'R')
info.run_procs++;
p->time_stamp = g_time;
}
void get_top_info(void)
{
DIR *dir;
struct dirent *entry;
if (!(dir = opendir("/proc"))) {
return;
}
info.run_procs = 0;
while ((entry = readdir(dir))) {
pid_t pid;
if (entry == NULL)
break;
if (sscanf(entry->d_name, "%u", &pid) != 1)
continue;
update_proc_entry(get_process(pid));
}
(void) closedir(dir);
}
/*
* Because Solaris systems often have 100s or 1000s of disks, we don't collect
* data for all of them but only for those mentioned in conkyrc.
* Instead of disk's special file in SVR4 format, we use the driver name and
* and the instance number to specify the disk or partition. For example: sd0,
* ssd3, or sd5,b.
*/
int update_diskio(void)
{
unsigned int tot_read = 0;
unsigned int tot_written = 0;
update_kstat();
for (struct diskio_stat *cur = &stats; cur; cur = cur->next) {
unsigned int read, written;
kstat_io_t *ksio;
kstat_t *ksp;
if (cur->dev == NULL)
continue;
if ((ksp = kstat_lookup(kstat, NULL, -1, cur->dev)) == NULL)
continue;
if (kstat_read(kstat, ksp, NULL) == -1)
continue;
ksio = (kstat_io_t *)ksp->ks_data;
tot_read += read = (unsigned int)(ksio->nread / 512);
tot_written += written = (unsigned int)(ksio->nwritten / 512);
update_diskio_values(cur, read, written);
}
update_diskio_values(&stats, tot_read, tot_written);
return 0;
}
void get_battery_short_status(char *buffer, unsigned int n, const char *bat)
{
/* Not implemented */
(void) bat;
if (buffer && n > 0)
memset(buffer, 0, n);
}
void get_acpi_fan(char *p_client_buffer, size_t client_buffer_size)
{
/* Not implemented */
if (p_client_buffer && client_buffer_size > 0)
memset(p_client_buffer, 0, client_buffer_size);
}
void get_acpi_ac_adapter(char *p_client_buffer, size_t client_buffer_size,
const char *adapter)
{
/* Not implemented */
if (p_client_buffer && client_buffer_size > 0)
memset(p_client_buffer, 0, client_buffer_size);
}
int get_battery_perct(const char *bat)
{
/* Not implemented */
(void) bat;
return 1;
}
int get_entropy_poolsize(unsigned int *val)
{
/* Not implemented */
(void) val;
return 1;
}
char get_freq(char *p_client_buffer, size_t client_buffer_size,
const char *p_format, int divisor, unsigned int cpu)
{
char stat_name[PATH_MAX];
kstat_named_t *knp;
snprintf(stat_name, PATH_MAX, "cpu_info%d", cpu - 1);
knp = get_kstat("cpu_info", cpu - 1, stat_name, "current_clock_Hz");
if (knp == NULL)
return 0;
snprintf(p_client_buffer, client_buffer_size, p_format,
(float) knp->value.ui32 / divisor / 1000000.0);
return 1;
}
int update_uptime(void)
{
kstat_named_t *knp;
knp = get_kstat("unix", -1, "system_misc", "boot_time");
if (knp == NULL)
return 0;
info.uptime = time(NULL) - knp->value.ui32;
return 1;
}
int open_acpi_temperature(const char *name)
{
/* Not implemented */
(void) name;
return 1;
}
int get_entropy_avail(unsigned int *val)
{
/* Not implemented */
(void) val;
return 1;
}
int update_load_average(void)
{
double load[3];
getloadavg(load, 3);
info.loadavg[0] = (float) load[0];
info.loadavg[1] = (float) load[1];
info.loadavg[2] = (float) load[2];
return 0;
}
void get_cpu_count(void)
{
kstat_named_t *knp = get_kstat("unix", -1, "system_misc", "ncpus");
if (knp != NULL)
info.cpu_count = knp->value.ui32;
}

8
src/solaris.h Normal file
View File

@ -0,0 +1,8 @@
#ifndef SOLARIS_H_
#define SOLARIS_H_
int get_entropy_avail(unsigned int *);
int get_entropy_poolsize(unsigned int *);
#endif /*SOLARIS_H_*/