#include <stdlib.h>
#include <string.h>
#include <dtrace.h>
#include <kstat.h>
#include <errno.h>
#include "powertop.h"
boolean_t g_turbo_supported;
double g_turbo_ratio;
static turbo_info_t *cpu_turbo_info = NULL;
static turbo_info_t *t_new = NULL;
static int
pt_turbo_init(void)
{
kstat_ctl_t *kc;
kstat_t *ksp;
kstat_named_t *knp;
if ((kc = kstat_open()) == NULL) {
g_turbo_supported = B_FALSE;
return (errno);
}
ksp = kstat_lookup(kc, "turbo", 0, NULL);
if (ksp == NULL) {
g_turbo_supported = B_FALSE;
(void) kstat_close(kc);
return (-1);
}
(void) kstat_read(kc, ksp, NULL);
knp = kstat_data_lookup(ksp, "turbo_supported");
if (knp == NULL) {
pt_error("couldn't find 'turbo_supported' kstat\n");
g_turbo_supported = B_FALSE;
(void) kstat_close(kc);
return (-2);
}
if (knp->value.ui32) {
g_turbo_supported = B_TRUE;
cpu_turbo_info = calloc((size_t)g_ncpus, sizeof (turbo_info_t));
t_new = calloc((size_t)g_ncpus, sizeof (turbo_info_t));
}
(void) kstat_close(kc);
return (0);
}
static int
pt_turbo_snapshot(turbo_info_t *turbo_snapshot)
{
kstat_ctl_t *kc;
kstat_t *ksp;
kstat_named_t *knp;
int cpu;
turbo_info_t *turbo_info;
if ((kc = kstat_open()) == NULL)
return (errno);
for (cpu = 0; cpu < g_ncpus; cpu++) {
turbo_info = &turbo_snapshot[cpu];
ksp = kstat_lookup(kc, "turbo", g_cpu_table[cpu], NULL);
if (ksp == NULL) {
pt_error("couldn't find 'turbo' kstat for CPU %d\n",
cpu);
(void) kstat_close(kc);
return (-1);
}
if (kstat_read(kc, ksp, NULL) == -1) {
pt_error("couldn't read 'turbo' kstat for CPU %d\n",
cpu);
(void) kstat_close(kc);
return (-2);
}
knp = kstat_data_lookup(ksp, "turbo_mcnt");
if (knp == NULL) {
pt_error("couldn't find 'turbo_mcnt' kstat for CPU "
"%d\n", cpu);
(void) kstat_close(kc);
return (-3);
}
turbo_info->t_mcnt = knp->value.ui64;
knp = kstat_data_lookup(ksp, "turbo_acnt");
if (knp == NULL) {
pt_error("couldn't find 'turbo_acnt' kstat for CPU "
"%d\n", cpu);
(void) kstat_close(kc);
return (-4);
}
turbo_info->t_acnt = knp->value.ui64;
}
if (kstat_close(kc) != 0)
pt_error("couldn't close 'turbo' kstat\n");
return (0);
}
int
pt_turbo_stat_prepare(void)
{
int ret = 0;
if ((ret = pt_turbo_init()) != 0)
return (ret);
if ((ret = pt_turbo_snapshot(cpu_turbo_info)) != 0)
pt_error("failed to snapshot 'turbo' kstat\n");
return (ret);
}
int
pt_turbo_stat_collect(void)
{
int cpu;
uint64_t delta_mcnt, delta_acnt;
double ratio;
int ret;
if ((ret = pt_turbo_snapshot(t_new)) != 0) {
pt_error("failed to snapshot 'turbo' kstat\n");
return (ret);
}
for (cpu = 0; cpu < g_ncpus; cpu++) {
delta_mcnt = t_new[cpu].t_mcnt - cpu_turbo_info[cpu].t_mcnt;
delta_acnt = t_new[cpu].t_acnt - cpu_turbo_info[cpu].t_acnt;
if ((delta_mcnt > delta_acnt) || (delta_mcnt == 0))
ratio = 1.0;
else
ratio = (double)delta_acnt / (double)delta_mcnt;
g_turbo_ratio += ratio;
}
g_turbo_ratio = g_turbo_ratio / (double)g_ncpus;
(void) memcpy(cpu_turbo_info, t_new, g_ncpus * (sizeof (turbo_info_t)));
return (0);
}