Blame src/graphing/dzl-cpu-model.c

rpm-build f53ec4
/* dzl-cpu-model.c
rpm-build f53ec4
 *
rpm-build f53ec4
 * Copyright (C) 2015 Christian Hergert <christian@hergert.me>
rpm-build f53ec4
 *
rpm-build f53ec4
 * This file is free software; you can redistribute it and/or modify it
rpm-build f53ec4
 * under the terms of the GNU Lesser General Public License as
rpm-build f53ec4
 * published by the Free Software Foundation; either version 3 of the
rpm-build f53ec4
 * License, or (at your option) any later version.
rpm-build f53ec4
 *
rpm-build f53ec4
 * This file is distributed in the hope that it will be useful, but
rpm-build f53ec4
 * WITHOUT ANY WARRANTY; without even the implied warranty of
rpm-build f53ec4
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
rpm-build f53ec4
 * Lesser General Public License for more details.
rpm-build f53ec4
 *
rpm-build f53ec4
 * You should have received a copy of the GNU General Public License
rpm-build f53ec4
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
rpm-build f53ec4
 */
rpm-build f53ec4
rpm-build f53ec4
#include "config.h"
rpm-build f53ec4
rpm-build f53ec4
#include <ctype.h>
rpm-build f53ec4
#include <fcntl.h>
rpm-build f53ec4
#include <stdlib.h>
rpm-build f53ec4
#include <unistd.h>
rpm-build f53ec4
#include <stdio.h>
rpm-build f53ec4
#if defined(__FreeBSD__)
rpm-build f53ec4
# include <errno.h>
rpm-build f53ec4
# include <sys/resource.h>
rpm-build f53ec4
# include <sys/sysctl.h>
rpm-build f53ec4
# include <sys/types.h>
rpm-build f53ec4
#endif
rpm-build f53ec4
rpm-build f53ec4
#include "graphing/dzl-cpu-model.h"
rpm-build f53ec4
#include "util/dzl-macros.h"
rpm-build f53ec4
rpm-build f53ec4
#ifdef __linux__
rpm-build f53ec4
# define PROC_STAT_BUF_SIZE 4096
rpm-build f53ec4
#endif
rpm-build f53ec4
rpm-build f53ec4
typedef struct
rpm-build f53ec4
{
rpm-build f53ec4
  gdouble total;
rpm-build f53ec4
  gdouble freq;
rpm-build f53ec4
  glong   last_user;
rpm-build f53ec4
  glong   last_idle;
rpm-build f53ec4
  glong   last_system;
rpm-build f53ec4
  glong   last_nice;
rpm-build f53ec4
  glong   last_iowait;
rpm-build f53ec4
  glong   last_irq;
rpm-build f53ec4
  glong   last_softirq;
rpm-build f53ec4
  glong   last_steal;
rpm-build f53ec4
  glong   last_guest;
rpm-build f53ec4
  glong   last_guest_nice;
rpm-build f53ec4
} CpuInfo;
rpm-build f53ec4
rpm-build f53ec4
struct _DzlCpuModel
rpm-build f53ec4
{
rpm-build f53ec4
  DzlGraphModel  parent_instance;
rpm-build f53ec4
rpm-build f53ec4
  GArray  *cpu_info;
rpm-build f53ec4
  guint    n_cpu;
rpm-build f53ec4
rpm-build f53ec4
#ifdef __linux__
rpm-build f53ec4
  gint     stat_fd;
rpm-build f53ec4
  gchar   *stat_buf;
rpm-build f53ec4
#endif
rpm-build f53ec4
rpm-build f53ec4
  guint    poll_source;
rpm-build f53ec4
  guint    poll_interval_msec;
rpm-build f53ec4
};
rpm-build f53ec4
rpm-build f53ec4
G_DEFINE_TYPE (DzlCpuModel, dzl_cpu_model, DZL_TYPE_GRAPH_MODEL)
rpm-build f53ec4
rpm-build f53ec4
#ifdef __linux__
rpm-build f53ec4
static gboolean
rpm-build f53ec4
read_stat (DzlCpuModel *self)
rpm-build f53ec4
{
rpm-build f53ec4
  gssize len;
rpm-build f53ec4
rpm-build f53ec4
  g_assert (self != NULL);
rpm-build f53ec4
  g_assert (self->stat_fd != -1);
rpm-build f53ec4
  g_assert (self->stat_buf != NULL);
rpm-build f53ec4
rpm-build f53ec4
  if (lseek (self->stat_fd, 0, SEEK_SET) != 0)
rpm-build f53ec4
    return FALSE;
rpm-build f53ec4
rpm-build f53ec4
  len = read (self->stat_fd, self->stat_buf, PROC_STAT_BUF_SIZE);
rpm-build f53ec4
  if (len <= 0)
rpm-build f53ec4
    return FALSE;
rpm-build f53ec4
rpm-build f53ec4
  if (len < PROC_STAT_BUF_SIZE)
rpm-build f53ec4
    self->stat_buf[len] = 0;
rpm-build f53ec4
  else
rpm-build f53ec4
    self->stat_buf[PROC_STAT_BUF_SIZE-1] = 0;
rpm-build f53ec4
rpm-build f53ec4
  return TRUE;
rpm-build f53ec4
}
rpm-build f53ec4
rpm-build f53ec4
static void
rpm-build f53ec4
dzl_cpu_model_poll (DzlCpuModel *self)
rpm-build f53ec4
{
rpm-build f53ec4
  gchar cpu[64] = { 0 };
rpm-build f53ec4
  glong user;
rpm-build f53ec4
  glong sys;
rpm-build f53ec4
  glong nice;
rpm-build f53ec4
  glong idle;
rpm-build f53ec4
  glong iowait;
rpm-build f53ec4
  glong irq;
rpm-build f53ec4
  glong softirq;
rpm-build f53ec4
  glong steal;
rpm-build f53ec4
  glong guest;
rpm-build f53ec4
  glong guest_nice;
rpm-build f53ec4
  glong user_calc;
rpm-build f53ec4
  glong system_calc;
rpm-build f53ec4
  glong nice_calc;
rpm-build f53ec4
  glong idle_calc;
rpm-build f53ec4
  glong iowait_calc;
rpm-build f53ec4
  glong irq_calc;
rpm-build f53ec4
  glong softirq_calc;
rpm-build f53ec4
  glong steal_calc;
rpm-build f53ec4
  glong guest_calc;
rpm-build f53ec4
  glong guest_nice_calc;
rpm-build f53ec4
  glong total;
rpm-build f53ec4
  gchar *line;
rpm-build f53ec4
  gint ret;
rpm-build f53ec4
  gint id;
rpm-build f53ec4
rpm-build f53ec4
  if (read_stat (self))
rpm-build f53ec4
    {
rpm-build f53ec4
      line = self->stat_buf;
rpm-build f53ec4
rpm-build f53ec4
      for (gsize i = 0; self->stat_buf[i]; i++)
rpm-build f53ec4
        {
rpm-build f53ec4
          if (self->stat_buf[i] == '\n') {
rpm-build f53ec4
            self->stat_buf[i] = '\0';
rpm-build f53ec4
rpm-build f53ec4
            if (strncmp (line, "cpu", 3) == 0)
rpm-build f53ec4
              {
rpm-build f53ec4
                if (isdigit (line[3]))
rpm-build f53ec4
                  {
rpm-build f53ec4
                    CpuInfo *cpu_info;
rpm-build f53ec4
rpm-build f53ec4
                    user = nice = sys = idle = id = 0;
rpm-build f53ec4
                    ret = sscanf (line, "%s %ld %ld %ld %ld %ld %ld %ld %ld %ld %ld",
rpm-build f53ec4
                                  cpu, &user, &nice, &sys, &idle,
rpm-build f53ec4
                                  &iowait, &irq, &softirq, &steal, &guest, &guest_nice);
rpm-build f53ec4
                    if (ret != 11)
rpm-build f53ec4
                      goto next;
rpm-build f53ec4
rpm-build f53ec4
                    ret = sscanf(cpu, "cpu%d", &id;;
rpm-build f53ec4
rpm-build f53ec4
                    if (ret != 1 || id < 0 || id >= (gint)self->n_cpu)
rpm-build f53ec4
                      goto next;
rpm-build f53ec4
rpm-build f53ec4
                    cpu_info = &g_array_index (self->cpu_info, CpuInfo, id);
rpm-build f53ec4
rpm-build f53ec4
                    user_calc = user - cpu_info->last_user;
rpm-build f53ec4
                    nice_calc = nice - cpu_info->last_nice;
rpm-build f53ec4
                    system_calc = sys - cpu_info->last_system;
rpm-build f53ec4
                    idle_calc = idle - cpu_info->last_idle;
rpm-build f53ec4
                    iowait_calc = iowait - cpu_info->last_iowait;
rpm-build f53ec4
                    irq_calc = irq - cpu_info->last_irq;
rpm-build f53ec4
                    softirq_calc = softirq - cpu_info->last_softirq;
rpm-build f53ec4
                    steal_calc = steal - cpu_info->last_steal;
rpm-build f53ec4
                    guest_calc = guest - cpu_info->last_guest;
rpm-build f53ec4
                    guest_nice_calc = guest_nice - cpu_info->last_guest_nice;
rpm-build f53ec4
rpm-build f53ec4
                    total = user_calc + nice_calc + system_calc + idle_calc + iowait_calc + irq_calc + softirq_calc + steal_calc + guest_calc + guest_nice_calc;
rpm-build f53ec4
                    cpu_info->total = ((total - idle_calc) / (gdouble)total) * 100.0;
rpm-build f53ec4
rpm-build f53ec4
                    cpu_info->last_user = user;
rpm-build f53ec4
                    cpu_info->last_nice = nice;
rpm-build f53ec4
                    cpu_info->last_idle = idle;
rpm-build f53ec4
                    cpu_info->last_system = sys;
rpm-build f53ec4
                    cpu_info->last_iowait = iowait;
rpm-build f53ec4
                    cpu_info->last_irq = irq;
rpm-build f53ec4
                    cpu_info->last_softirq = softirq;
rpm-build f53ec4
                    cpu_info->last_steal = steal;
rpm-build f53ec4
                    cpu_info->last_guest = guest;
rpm-build f53ec4
                    cpu_info->last_guest_nice = guest_nice;
rpm-build f53ec4
                  }
rpm-build f53ec4
              } else {
rpm-build f53ec4
                /* CPU info comes first. Skip further lines. */
rpm-build f53ec4
                break;
rpm-build f53ec4
              }
rpm-build f53ec4
rpm-build f53ec4
            next:
rpm-build f53ec4
              line = &self->stat_buf[i + 1];
rpm-build f53ec4
            }
rpm-build f53ec4
        }
rpm-build f53ec4
    }
rpm-build f53ec4
}
rpm-build f53ec4
rpm-build f53ec4
#elif defined(__FreeBSD__)
rpm-build f53ec4
rpm-build f53ec4
static void
rpm-build f53ec4
dzl_cpu_model_poll (DzlCpuModel *self)
rpm-build f53ec4
{
rpm-build f53ec4
  static gint mib_cp_times[2];
rpm-build f53ec4
  static gsize len_cp_times = 2;
rpm-build f53ec4
rpm-build f53ec4
  if (mib_cp_times[0] == 0 || mib_cp_times[1] == 0)
rpm-build f53ec4
    {
rpm-build f53ec4
      if (sysctlnametomib ("kern.cp_times", mib_cp_times, &len_cp_times) == -1)
rpm-build f53ec4
        {
rpm-build f53ec4
          g_critical ("Cannot convert sysctl name kern.cp_times to a mib array: %s",
rpm-build f53ec4
                      g_strerror (errno));
rpm-build f53ec4
          return;
rpm-build f53ec4
        }
rpm-build f53ec4
    }
rpm-build f53ec4
rpm-build f53ec4
  gsize cp_times_size = sizeof (glong) * CPUSTATES * self->n_cpu;
rpm-build f53ec4
  glong *cp_times = g_malloc (cp_times_size);
rpm-build f53ec4
rpm-build f53ec4
  if (sysctl (mib_cp_times, 2, cp_times, &cp_times_size, NULL, 0) == -1)
rpm-build f53ec4
    {
rpm-build f53ec4
      g_critical ("Cannot get CPU usage by sysctl kern.cp_times: %s",
rpm-build f53ec4
                  g_strerror (errno));
rpm-build f53ec4
      g_free (cp_times);
rpm-build f53ec4
      return;
rpm-build f53ec4
    }
rpm-build f53ec4
rpm-build f53ec4
  for (guint i = 0, j = 0; i < self->n_cpu; i++, j += CPUSTATES)
rpm-build f53ec4
    {
rpm-build f53ec4
      CpuInfo *cpu_info = &g_array_index (self->cpu_info, CpuInfo, i);
rpm-build f53ec4
rpm-build f53ec4
      glong user = cp_times[j + CP_USER];
rpm-build f53ec4
      glong nice = cp_times[j + CP_NICE];
rpm-build f53ec4
      glong sys = cp_times[j + CP_SYS];
rpm-build f53ec4
      glong irq = cp_times[j + CP_INTR];
rpm-build f53ec4
      glong idle = cp_times[j + CP_IDLE];
rpm-build f53ec4
rpm-build f53ec4
      glong user_calc = user - cpu_info->last_user;
rpm-build f53ec4
      glong nice_calc = nice - cpu_info->last_nice;
rpm-build f53ec4
      glong system_calc = sys - cpu_info->last_system;
rpm-build f53ec4
      glong irq_calc = irq - cpu_info->last_irq;
rpm-build f53ec4
      glong idle_calc = idle - cpu_info->last_idle;
rpm-build f53ec4
rpm-build f53ec4
      glong total = user_calc + nice_calc + system_calc + irq_calc + idle_calc;
rpm-build f53ec4
      cpu_info->total = ((total - idle_calc) / (gdouble)total) * 100.0;
rpm-build f53ec4
rpm-build f53ec4
      cpu_info->last_user = user;
rpm-build f53ec4
      cpu_info->last_nice = nice;
rpm-build f53ec4
      cpu_info->last_system = sys;
rpm-build f53ec4
      cpu_info->last_irq = irq;
rpm-build f53ec4
      cpu_info->last_idle = idle;
rpm-build f53ec4
    }
rpm-build f53ec4
  g_free (cp_times);
rpm-build f53ec4
}
rpm-build f53ec4
#else
rpm-build f53ec4
static void
rpm-build f53ec4
dzl_cpu_model_poll (DzlCpuModel *self)
rpm-build f53ec4
{
rpm-build f53ec4
  /*
rpm-build f53ec4
   * TODO: calculate cpu info for OpenBSD/etc.
rpm-build f53ec4
   *
rpm-build f53ec4
   * While we are at it, we should make the Linux code above non-shitty.
rpm-build f53ec4
   */
rpm-build f53ec4
}
rpm-build f53ec4
#endif
rpm-build f53ec4
rpm-build f53ec4
static gboolean
rpm-build f53ec4
dzl_cpu_model_poll_cb (gpointer user_data)
rpm-build f53ec4
{
rpm-build f53ec4
  DzlCpuModel *self = user_data;
rpm-build f53ec4
  DzlGraphModelIter iter;
rpm-build f53ec4
  guint i;
rpm-build f53ec4
rpm-build f53ec4
  dzl_cpu_model_poll (self);
rpm-build f53ec4
rpm-build f53ec4
  dzl_graph_view_model_push (DZL_GRAPH_MODEL (self), &iter, g_get_monotonic_time ());
rpm-build f53ec4
rpm-build f53ec4
  for (i = 0; i < self->cpu_info->len; i++)
rpm-build f53ec4
    {
rpm-build f53ec4
      CpuInfo *cpu_info;
rpm-build f53ec4
rpm-build f53ec4
      cpu_info = &g_array_index (self->cpu_info, CpuInfo, i);
rpm-build f53ec4
      dzl_graph_view_model_iter_set (&iter, i, cpu_info->total, -1);
rpm-build f53ec4
    }
rpm-build f53ec4
rpm-build f53ec4
  return G_SOURCE_CONTINUE;
rpm-build f53ec4
}
rpm-build f53ec4
rpm-build f53ec4
static void
rpm-build f53ec4
dzl_cpu_model_constructed (GObject *object)
rpm-build f53ec4
{
rpm-build f53ec4
  DzlCpuModel *self = (DzlCpuModel *)object;
rpm-build f53ec4
  gint64 timespan;
rpm-build f53ec4
  guint max_samples;
rpm-build f53ec4
rpm-build f53ec4
  G_OBJECT_CLASS (dzl_cpu_model_parent_class)->constructed (object);
rpm-build f53ec4
rpm-build f53ec4
  max_samples = dzl_graph_view_model_get_max_samples (DZL_GRAPH_MODEL (self));
rpm-build f53ec4
  timespan = dzl_graph_view_model_get_timespan (DZL_GRAPH_MODEL (self));
rpm-build f53ec4
rpm-build f53ec4
  self->poll_interval_msec = (gdouble)timespan / (gdouble)(max_samples - 1) / 1000L;
rpm-build f53ec4
rpm-build f53ec4
  if (self->poll_interval_msec == 0)
rpm-build f53ec4
    {
rpm-build f53ec4
      g_critical ("Implausible timespan/max_samples combination for graph.");
rpm-build f53ec4
      self->poll_interval_msec = 1000;
rpm-build f53ec4
    }
rpm-build f53ec4
rpm-build f53ec4
  self->n_cpu = g_get_num_processors ();
rpm-build f53ec4
rpm-build f53ec4
  for (guint i = 0; i < self->n_cpu; i++)
rpm-build f53ec4
    {
rpm-build f53ec4
      CpuInfo cpu_info = { 0 };
rpm-build f53ec4
      DzlGraphColumn *column;
rpm-build f53ec4
      gchar *name;
rpm-build f53ec4
rpm-build f53ec4
      name = g_strdup_printf ("CPU %d", i + 1);
rpm-build f53ec4
      column = dzl_graph_view_column_new (name, G_TYPE_DOUBLE);
rpm-build f53ec4
rpm-build f53ec4
      dzl_graph_view_model_add_column (DZL_GRAPH_MODEL (self), column);
rpm-build f53ec4
      g_array_append_val (self->cpu_info, cpu_info);
rpm-build f53ec4
rpm-build f53ec4
      g_object_unref (column);
rpm-build f53ec4
      g_free (name);
rpm-build f53ec4
    }
rpm-build f53ec4
rpm-build f53ec4
  dzl_cpu_model_poll (self);
rpm-build f53ec4
rpm-build f53ec4
  self->poll_source = g_timeout_add (self->poll_interval_msec, dzl_cpu_model_poll_cb, self);
rpm-build f53ec4
}
rpm-build f53ec4
rpm-build f53ec4
static void
rpm-build f53ec4
dzl_cpu_model_finalize (GObject *object)
rpm-build f53ec4
{
rpm-build f53ec4
  DzlCpuModel *self = (DzlCpuModel *)object;
rpm-build f53ec4
rpm-build f53ec4
#ifdef __linux__
rpm-build f53ec4
  g_clear_pointer (&self->stat_buf, g_free);
rpm-build f53ec4
  if (self->stat_fd != -1)
rpm-build f53ec4
    close (self->stat_fd);
rpm-build f53ec4
#endif
rpm-build f53ec4
rpm-build f53ec4
  dzl_clear_source (&self->poll_source);
rpm-build f53ec4
  g_clear_pointer (&self->cpu_info, g_array_unref);
rpm-build f53ec4
rpm-build f53ec4
  G_OBJECT_CLASS (dzl_cpu_model_parent_class)->finalize (object);
rpm-build f53ec4
}
rpm-build f53ec4
rpm-build f53ec4
static void
rpm-build f53ec4
dzl_cpu_model_class_init (DzlCpuModelClass *klass)
rpm-build f53ec4
{
rpm-build f53ec4
  GObjectClass *object_class = G_OBJECT_CLASS (klass);
rpm-build f53ec4
rpm-build f53ec4
  object_class->constructed = dzl_cpu_model_constructed;
rpm-build f53ec4
  object_class->finalize = dzl_cpu_model_finalize;
rpm-build f53ec4
}
rpm-build f53ec4
rpm-build f53ec4
static void
rpm-build f53ec4
dzl_cpu_model_init (DzlCpuModel *self)
rpm-build f53ec4
{
rpm-build f53ec4
  self->cpu_info = g_array_new (FALSE, FALSE, sizeof (CpuInfo));
rpm-build f53ec4
rpm-build f53ec4
#ifdef __linux__
rpm-build f53ec4
  self->stat_fd = open ("/proc/stat", O_RDONLY);
rpm-build f53ec4
  self->stat_buf = g_malloc (PROC_STAT_BUF_SIZE);
rpm-build f53ec4
#endif
rpm-build f53ec4
rpm-build f53ec4
  g_object_set (self,
rpm-build f53ec4
                "value-min", 0.0,
rpm-build f53ec4
                "value-max", 100.0,
rpm-build f53ec4
                NULL);
rpm-build f53ec4
}
rpm-build f53ec4
rpm-build f53ec4
DzlGraphModel *
rpm-build f53ec4
dzl_cpu_model_new (void)
rpm-build f53ec4
{
rpm-build f53ec4
  return g_object_new (DZL_TYPE_CPU_MODEL, NULL);
rpm-build f53ec4
}