2022-07-11 02:50:39 -04:00
|
|
|
local compound_dial = require 'compound_dial'
|
2021-07-29 22:37:30 -04:00
|
|
|
local line = require 'line'
|
2021-08-08 18:19:37 -04:00
|
|
|
local text_table = require 'text_table'
|
2021-08-08 15:58:53 -04:00
|
|
|
local i_o = require 'i_o'
|
2021-07-29 22:18:29 -04:00
|
|
|
local common = require 'common'
|
|
|
|
local geometry = require 'geometry'
|
2021-08-08 15:10:09 -04:00
|
|
|
local cpu = require 'sys'
|
|
|
|
local pure = require 'pure'
|
2019-09-01 15:34:10 -04:00
|
|
|
|
2021-07-13 01:01:21 -04:00
|
|
|
local __math_floor = math.floor
|
2021-07-12 23:49:52 -04:00
|
|
|
|
2021-07-17 23:06:30 -04:00
|
|
|
return function(update_freq)
|
2022-07-11 02:50:39 -04:00
|
|
|
-- local SHOW_DIALS = true
|
|
|
|
-- local SHOW_TIMESERIES = true
|
|
|
|
-- local SHOW_TABLE = true
|
|
|
|
|
2021-07-17 23:06:30 -04:00
|
|
|
local MODULE_Y = 614
|
|
|
|
local DIAL_INNER_RADIUS = 30
|
|
|
|
local DIAL_OUTER_RADIUS = 42
|
|
|
|
local DIAL_THICKNESS = 5.5
|
|
|
|
local SEPARATOR_SPACING = 20
|
|
|
|
local TEXT_SPACING = 22
|
|
|
|
local PLOT_SECTION_BREAK = 23
|
|
|
|
local PLOT_HEIGHT = 56
|
|
|
|
local TABLE_SECTION_BREAK = 20
|
|
|
|
local TABLE_HEIGHT = 114
|
|
|
|
|
|
|
|
-----------------------------------------------------------------------------
|
|
|
|
-- header
|
|
|
|
|
2022-07-13 01:06:15 -04:00
|
|
|
local mk_header = pure.partial(
|
|
|
|
common.mk_header,
|
|
|
|
'PROCESSOR',
|
|
|
|
geometry.SECTION_WIDTH,
|
|
|
|
geometry.LEFT_X
|
|
|
|
)
|
2017-07-19 00:36:15 -04:00
|
|
|
|
2021-07-17 23:06:30 -04:00
|
|
|
-----------------------------------------------------------------------------
|
|
|
|
-- cores (loads and temps)
|
2021-07-17 16:43:00 -04:00
|
|
|
|
2022-07-11 02:50:39 -04:00
|
|
|
-- this totally is not supposed to be a state monad (ssssh...)
|
|
|
|
local update_state = function(trigger, cpu_loads)
|
|
|
|
return {
|
|
|
|
cpu_loads = cpu.read_cpu_loads(cpu_loads),
|
|
|
|
load_sum = 0,
|
|
|
|
trigger = trigger
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
local state = update_state(0, cpu.init_cpu_loads())
|
2021-07-29 22:39:38 -04:00
|
|
|
local ncpus = cpu.get_cpu_number()
|
|
|
|
local ncores = cpu.get_core_number()
|
2021-07-17 21:22:17 -04:00
|
|
|
local nthreads = ncpus / ncores
|
|
|
|
|
2021-07-17 23:06:30 -04:00
|
|
|
local create_core = function(x, y)
|
|
|
|
return {
|
2021-07-30 22:02:46 -04:00
|
|
|
loads = common.make_compound_dial(
|
2021-07-17 23:06:30 -04:00
|
|
|
x,
|
|
|
|
y,
|
|
|
|
DIAL_OUTER_RADIUS,
|
|
|
|
DIAL_INNER_RADIUS,
|
|
|
|
DIAL_THICKNESS,
|
2021-11-10 12:24:13 -05:00
|
|
|
80,
|
2021-07-17 23:06:30 -04:00
|
|
|
nthreads
|
|
|
|
),
|
2021-07-29 23:04:39 -04:00
|
|
|
coretemp = common.make_text_circle(
|
2021-07-17 23:06:30 -04:00
|
|
|
x,
|
|
|
|
y,
|
|
|
|
DIAL_INNER_RADIUS - 2,
|
|
|
|
'%s°C',
|
2021-11-10 16:26:48 -05:00
|
|
|
80,
|
|
|
|
__math_floor
|
2021-07-17 23:06:30 -04:00
|
|
|
)
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2022-07-11 02:50:39 -04:00
|
|
|
local mk_cores = function(y)
|
|
|
|
local coretemp_paths = cpu.get_coretemp_paths()
|
|
|
|
local cores = {}
|
2022-07-12 01:23:59 -04:00
|
|
|
-- TODO what happens when the number of cores changes?
|
2022-07-11 02:50:39 -04:00
|
|
|
for c = 1, ncores do
|
|
|
|
local dial_x = geometry.LEFT_X + DIAL_OUTER_RADIUS +
|
|
|
|
(geometry.SECTION_WIDTH - 2 * DIAL_OUTER_RADIUS) * (c - 1) / 3
|
|
|
|
local dial_y = y + DIAL_OUTER_RADIUS
|
|
|
|
cores[c] = create_core(dial_x, dial_y)
|
|
|
|
end
|
|
|
|
local update = function(state_)
|
|
|
|
local s = state_.load_sum
|
|
|
|
for _, load_data in pairs(state_.cpu_loads) do
|
|
|
|
local cur = load_data.percent_active
|
|
|
|
s = s + cur
|
|
|
|
compound_dial.set(
|
|
|
|
cores[load_data.conky_core_id].loads,
|
|
|
|
load_data.conky_thread_id,
|
|
|
|
cur * 100
|
|
|
|
)
|
|
|
|
end
|
|
|
|
for conky_core_id, path in pairs(coretemp_paths) do
|
|
|
|
local temp = __math_floor(0.001 * i_o.read_file(path, nil, '*n'))
|
|
|
|
common.text_circle_set(cores[conky_core_id].coretemp, temp)
|
|
|
|
end
|
|
|
|
state_.load_sum = s
|
|
|
|
return state_
|
|
|
|
end
|
|
|
|
local static = function(cr)
|
|
|
|
for i = 1, #cores do
|
|
|
|
common.text_circle_draw_static(cores[i].coretemp, cr)
|
|
|
|
compound_dial.draw_static(cores[i].loads, cr)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
local dynamic = function(cr)
|
|
|
|
for i = 1, #cores do
|
|
|
|
common.text_circle_draw_dynamic(cores[i].coretemp, cr)
|
2022-07-12 01:23:59 -04:00
|
|
|
compound_dial.draw_dynamic(cores[i].loads, cr)
|
2022-07-11 02:50:39 -04:00
|
|
|
end
|
|
|
|
end
|
2022-07-12 01:23:59 -04:00
|
|
|
return common.mk_acc(DIAL_OUTER_RADIUS * 2, update, static, dynamic)
|
2021-07-17 21:22:17 -04:00
|
|
|
end
|
2021-07-17 16:43:00 -04:00
|
|
|
|
2021-07-17 23:06:30 -04:00
|
|
|
-----------------------------------------------------------------------------
|
|
|
|
-- HWP status
|
|
|
|
|
2022-07-11 02:50:39 -04:00
|
|
|
local mk_hwp_freq = function(y)
|
|
|
|
local hwp_paths = cpu.get_hwp_paths()
|
|
|
|
local cpu_status = common.make_text_rows(
|
|
|
|
geometry.LEFT_X,
|
|
|
|
y,
|
|
|
|
geometry.SECTION_WIDTH,
|
|
|
|
TEXT_SPACING,
|
|
|
|
{'HWP Preference', 'Ave Freq'}
|
|
|
|
)
|
|
|
|
local update = function(state_)
|
|
|
|
-- For some reason this call is slow (querying anything with pstate in
|
|
|
|
-- general seems slow), but I also don't need to see an update every
|
|
|
|
-- cycle, hence the trigger
|
|
|
|
if state_.trigger == 0 then
|
|
|
|
common.text_rows_set(cpu_status, 1, cpu.read_hwp(hwp_paths))
|
|
|
|
end
|
|
|
|
common.text_rows_set(cpu_status, 2, cpu.read_freq())
|
|
|
|
return state_
|
|
|
|
end
|
|
|
|
local static = pure.partial(common.text_rows_draw_static, cpu_status)
|
|
|
|
local dynamic = pure.partial(common.text_rows_draw_dynamic, cpu_status)
|
2022-07-12 01:23:59 -04:00
|
|
|
return common.mk_acc(TEXT_SPACING, update, static, dynamic)
|
2022-07-11 02:50:39 -04:00
|
|
|
end
|
2021-07-17 23:06:30 -04:00
|
|
|
|
|
|
|
-----------------------------------------------------------------------------
|
|
|
|
-- frequency
|
|
|
|
|
2022-07-13 01:06:15 -04:00
|
|
|
local mk_sep = pure.partial(
|
|
|
|
common.mk_seperator,
|
|
|
|
geometry.SECTION_WIDTH,
|
|
|
|
geometry.LEFT_X
|
|
|
|
)
|
2021-07-17 23:06:30 -04:00
|
|
|
|
|
|
|
-----------------------------------------------------------------------------
|
|
|
|
-- total load plot
|
|
|
|
|
2022-07-11 02:50:39 -04:00
|
|
|
local mk_load_plot = function(y)
|
|
|
|
local total_load = common.make_tagged_percent_timeseries(
|
|
|
|
geometry.LEFT_X,
|
|
|
|
y,
|
|
|
|
geometry.SECTION_WIDTH,
|
|
|
|
PLOT_HEIGHT,
|
|
|
|
PLOT_SECTION_BREAK,
|
|
|
|
"Total Load",
|
|
|
|
update_freq
|
|
|
|
)
|
|
|
|
local update = function(state_)
|
|
|
|
common.tagged_percent_timeseries_set(
|
|
|
|
total_load,
|
|
|
|
state_.load_sum / ncpus * 100
|
|
|
|
)
|
|
|
|
return state_
|
|
|
|
end
|
|
|
|
local static = pure.partial(common.tagged_percent_timeseries_draw_static, total_load)
|
|
|
|
local dynamic = pure.partial(common.tagged_percent_timeseries_draw_dynamic, total_load)
|
2022-07-12 01:23:59 -04:00
|
|
|
return common.mk_acc(PLOT_HEIGHT + PLOT_SECTION_BREAK, update, static, dynamic)
|
2022-07-11 02:50:39 -04:00
|
|
|
end
|
2021-07-17 23:06:30 -04:00
|
|
|
|
|
|
|
-----------------------------------------------------------------------------
|
|
|
|
-- cpu top table
|
|
|
|
|
2022-07-11 02:50:39 -04:00
|
|
|
local mk_tbl = function(y)
|
|
|
|
local NUM_ROWS = 5
|
|
|
|
local TABLE_CONKY = pure.map_n(
|
|
|
|
function(i) return {pid = '${top pid '..i..'}', cpu = '${top cpu '..i..'}'} end,
|
|
|
|
NUM_ROWS
|
|
|
|
)
|
|
|
|
local tbl = common.make_text_table(
|
|
|
|
geometry.LEFT_X,
|
|
|
|
y,
|
|
|
|
geometry.SECTION_WIDTH,
|
|
|
|
TABLE_HEIGHT,
|
|
|
|
NUM_ROWS,
|
|
|
|
'CPU (%)'
|
|
|
|
)
|
|
|
|
local update = function(state_)
|
|
|
|
for r = 1, NUM_ROWS do
|
|
|
|
local pid = i_o.conky(TABLE_CONKY[r].pid, '(%d+)') -- may have leading spaces
|
|
|
|
if pid ~= '' then
|
|
|
|
text_table.set(tbl, 1, r, i_o.read_file('/proc/'..pid..'/comm', '(%C+)'))
|
|
|
|
text_table.set(tbl, 2, r, pid)
|
|
|
|
text_table.set(tbl, 3, r, i_o.conky(TABLE_CONKY[r].cpu))
|
|
|
|
end
|
2021-07-17 16:43:00 -04:00
|
|
|
end
|
2022-07-11 02:50:39 -04:00
|
|
|
return state_
|
2019-09-01 15:34:10 -04:00
|
|
|
end
|
2022-07-11 02:50:39 -04:00
|
|
|
local static = pure.partial(text_table.draw_static, tbl)
|
|
|
|
local dynamic = pure.partial(text_table.draw_dynamic, tbl)
|
2022-07-12 01:23:59 -04:00
|
|
|
return common.mk_acc(TABLE_HEIGHT, update, static, dynamic)
|
2018-08-05 11:35:00 -04:00
|
|
|
end
|
2018-08-05 11:22:07 -04:00
|
|
|
|
2022-07-11 02:50:39 -04:00
|
|
|
-----------------------------------------------------------------------------
|
|
|
|
-- main functions
|
2018-08-05 11:35:00 -04:00
|
|
|
|
2022-07-12 01:23:59 -04:00
|
|
|
local rbs = common.reduce_blocks(
|
|
|
|
MODULE_Y,
|
2022-07-11 02:50:39 -04:00
|
|
|
{
|
2022-07-12 01:23:59 -04:00
|
|
|
common.mk_block(mk_header, true, 0),
|
|
|
|
common.mk_block(mk_cores, true, 0),
|
|
|
|
common.mk_block(mk_hwp_freq, true, TEXT_SPACING),
|
|
|
|
common.mk_block(mk_sep, true, SEPARATOR_SPACING),
|
|
|
|
common.mk_block(mk_load_plot, true, SEPARATOR_SPACING),
|
|
|
|
common.mk_block(mk_tbl, true, TABLE_SECTION_BREAK)
|
2022-07-11 02:50:39 -04:00
|
|
|
}
|
|
|
|
)
|
2018-08-05 11:22:07 -04:00
|
|
|
|
2022-07-11 02:50:39 -04:00
|
|
|
local update = function(trigger)
|
2022-07-12 01:23:59 -04:00
|
|
|
rbs.updater(update_state(trigger, state.cpu_loads))
|
2018-08-05 11:08:37 -04:00
|
|
|
end
|
2018-01-05 23:36:50 -05:00
|
|
|
|
2022-07-12 01:23:59 -04:00
|
|
|
-- TODO return the bottom y/height of the entire module
|
|
|
|
return {
|
|
|
|
static = rbs.static_drawer,
|
|
|
|
dynamic = rbs.dynamic_drawer,
|
|
|
|
update = update
|
|
|
|
}
|
2021-07-17 00:17:22 -04:00
|
|
|
end
|