tools/lib/thermal: Add a thermal library
The thermal framework implements a netlink notification mechanism to be used by the userspace to have a thermal configuration discovery, trip point changes or violation, cooling device changes notifications, etc... This library provides a level of abstraction for the thermal netlink notification allowing the userspace to connect to the notification mechanism more easily. The library is callback oriented. Signed-off-by: Daniel Lezcano <daniel.lezcano@linaro.org> Tested-by: Srinivas Pandruvada <srinivas.pandruvada@linux.intel.com> Link: https://lore.kernel.org/r/20220420160933.347088-2-daniel.lezcano@linaro.org
This commit is contained in:
parent
bf70c57751
commit
47c4b0de08
|
@ -19541,6 +19541,7 @@ F: drivers/thermal/
|
|||
F: include/linux/cpu_cooling.h
|
||||
F: include/linux/thermal.h
|
||||
F: include/uapi/linux/thermal.h
|
||||
F: tools/lib/thermal/
|
||||
F: tools/thermal/
|
||||
|
||||
THERMAL DRIVER FOR AMLOGIC SOCS
|
||||
|
|
|
@ -31,6 +31,7 @@ help:
|
|||
@echo ' bootconfig - boot config tool'
|
||||
@echo ' spi - spi tools'
|
||||
@echo ' tmon - thermal monitoring and tuning tool'
|
||||
@echo ' thermal - thermal library'
|
||||
@echo ' tracing - misc tracing tools'
|
||||
@echo ' turbostat - Intel CPU idle stats and freq reporting tool'
|
||||
@echo ' usb - USB testing tools'
|
||||
|
@ -85,6 +86,9 @@ perf: FORCE
|
|||
selftests: FORCE
|
||||
$(call descend,testing/$@)
|
||||
|
||||
thermal: FORCE
|
||||
$(call descend,lib/$@)
|
||||
|
||||
turbostat x86_energy_perf_policy intel-speed-select: FORCE
|
||||
$(call descend,power/x86/$@)
|
||||
|
||||
|
@ -101,7 +105,7 @@ all: acpi cgroup counter cpupower gpio hv firewire \
|
|||
perf selftests bootconfig spi turbostat usb \
|
||||
virtio vm bpf x86_energy_perf_policy \
|
||||
tmon freefall iio objtool kvm_stat wmi \
|
||||
pci debugging tracing
|
||||
pci debugging tracing thermal
|
||||
|
||||
acpi_install:
|
||||
$(call descend,power/$(@:_install=),install)
|
||||
|
@ -115,6 +119,9 @@ cgroup_install counter_install firewire_install gpio_install hv_install iio_inst
|
|||
selftests_install:
|
||||
$(call descend,testing/$(@:_install=),install)
|
||||
|
||||
thermal_install:
|
||||
$(call descend,lib/$(@:_install=),install)
|
||||
|
||||
turbostat_install x86_energy_perf_policy_install intel-speed-select_install:
|
||||
$(call descend,power/x86/$(@:_install=),install)
|
||||
|
||||
|
@ -160,6 +167,9 @@ perf_clean:
|
|||
selftests_clean:
|
||||
$(call descend,testing/$(@:_clean=),clean)
|
||||
|
||||
thermal_clean:
|
||||
$(call descend,lib/thermal,clean)
|
||||
|
||||
turbostat_clean x86_energy_perf_policy_clean intel-speed-select_clean:
|
||||
$(call descend,power/x86/$(@:_clean=),clean)
|
||||
|
||||
|
@ -177,6 +187,6 @@ clean: acpi_clean cgroup_clean counter_clean cpupower_clean hv_clean firewire_cl
|
|||
vm_clean bpf_clean iio_clean x86_energy_perf_policy_clean tmon_clean \
|
||||
freefall_clean build_clean libbpf_clean libsubcmd_clean \
|
||||
gpio_clean objtool_clean leds_clean wmi_clean pci_clean firmware_clean debugging_clean \
|
||||
intel-speed-select_clean tracing_clean
|
||||
intel-speed-select_clean tracing_clean thermal_clean
|
||||
|
||||
.PHONY: FORCE
|
||||
|
|
|
@ -0,0 +1,2 @@
|
|||
libthermal.so*
|
||||
libthermal.pc
|
|
@ -0,0 +1,5 @@
|
|||
libthermal-y += commands.o
|
||||
libthermal-y += events.o
|
||||
libthermal-y += thermal_nl.o
|
||||
libthermal-y += sampling.o
|
||||
libthermal-y += thermal.o
|
|
@ -0,0 +1,165 @@
|
|||
# SPDX-License-Identifier: (LGPL-2.1 OR BSD-2-Clause)
|
||||
# Most of this file is copied from tools/lib/perf/Makefile
|
||||
|
||||
LIBTHERMAL_VERSION = 0
|
||||
LIBTHERMAL_PATCHLEVEL = 0
|
||||
LIBTHERMAL_EXTRAVERSION = 1
|
||||
|
||||
MAKEFLAGS += --no-print-directory
|
||||
|
||||
ifeq ($(srctree),)
|
||||
srctree := $(patsubst %/,%,$(dir $(CURDIR)))
|
||||
srctree := $(patsubst %/,%,$(dir $(srctree)))
|
||||
srctree := $(patsubst %/,%,$(dir $(srctree)))
|
||||
# $(info Determined 'srctree' to be $(srctree))
|
||||
endif
|
||||
|
||||
INSTALL = install
|
||||
|
||||
# Use DESTDIR for installing into a different root directory.
|
||||
# This is useful for building a package. The program will be
|
||||
# installed in this directory as if it was the root directory.
|
||||
# Then the build tool can move it later.
|
||||
DESTDIR ?=
|
||||
DESTDIR_SQ = '$(subst ','\'',$(DESTDIR))'
|
||||
|
||||
include $(srctree)/tools/scripts/Makefile.include
|
||||
include $(srctree)/tools/scripts/Makefile.arch
|
||||
|
||||
ifeq ($(LP64), 1)
|
||||
libdir_relative = lib64
|
||||
else
|
||||
libdir_relative = lib
|
||||
endif
|
||||
|
||||
prefix ?=
|
||||
libdir = $(prefix)/$(libdir_relative)
|
||||
|
||||
# Shell quotes
|
||||
libdir_SQ = $(subst ','\'',$(libdir))
|
||||
libdir_relative_SQ = $(subst ','\'',$(libdir_relative))
|
||||
|
||||
ifeq ("$(origin V)", "command line")
|
||||
VERBOSE = $(V)
|
||||
endif
|
||||
ifndef VERBOSE
|
||||
VERBOSE = 0
|
||||
endif
|
||||
|
||||
ifeq ($(VERBOSE),1)
|
||||
Q =
|
||||
else
|
||||
Q = @
|
||||
endif
|
||||
|
||||
# Set compile option CFLAGS
|
||||
ifdef EXTRA_CFLAGS
|
||||
CFLAGS := $(EXTRA_CFLAGS)
|
||||
else
|
||||
CFLAGS := -g -Wall
|
||||
endif
|
||||
|
||||
INCLUDES = \
|
||||
-I/usr/include/libnl3 \
|
||||
-I$(srctree)/tools/lib/thermal/include \
|
||||
-I$(srctree)/tools/lib/ \
|
||||
-I$(srctree)/tools/include \
|
||||
-I$(srctree)/tools/arch/$(SRCARCH)/include/ \
|
||||
-I$(srctree)/tools/arch/$(SRCARCH)/include/uapi \
|
||||
-I$(srctree)/tools/include/uapi
|
||||
|
||||
# Append required CFLAGS
|
||||
override CFLAGS += $(EXTRA_WARNINGS)
|
||||
override CFLAGS += -Werror -Wall
|
||||
override CFLAGS += -fPIC
|
||||
override CFLAGS += $(INCLUDES)
|
||||
override CFLAGS += -fvisibility=hidden
|
||||
override CFGLAS += -Wl,-L.
|
||||
override CFGLAS += -Wl,-lthermal
|
||||
|
||||
all:
|
||||
|
||||
export srctree OUTPUT CC LD CFLAGS V
|
||||
export DESTDIR DESTDIR_SQ
|
||||
|
||||
include $(srctree)/tools/build/Makefile.include
|
||||
|
||||
VERSION_SCRIPT := libthermal.map
|
||||
|
||||
PATCHLEVEL = $(LIBTHERMAL_PATCHLEVEL)
|
||||
EXTRAVERSION = $(LIBTHERMAL_EXTRAVERSION)
|
||||
VERSION = $(LIBTHERMAL_VERSION).$(LIBTHERMAL_PATCHLEVEL).$(LIBTHERMAL_EXTRAVERSION)
|
||||
|
||||
LIBTHERMAL_SO := $(OUTPUT)libthermal.so.$(VERSION)
|
||||
LIBTHERMAL_A := $(OUTPUT)libthermal.a
|
||||
LIBTHERMAL_IN := $(OUTPUT)libthermal-in.o
|
||||
LIBTHERMAL_PC := $(OUTPUT)libthermal.pc
|
||||
LIBTHERMAL_ALL := $(LIBTHERMAL_A) $(OUTPUT)libthermal.so*
|
||||
|
||||
THERMAL_UAPI := include/uapi/linux/thermal.h
|
||||
|
||||
$(THERMAL_UAPI): FORCE
|
||||
ln -sf $(srctree)/$@ $(srctree)/tools/$@
|
||||
|
||||
$(LIBTHERMAL_IN): FORCE
|
||||
$(Q)$(MAKE) $(build)=libthermal
|
||||
|
||||
$(LIBTHERMAL_A): $(LIBTHERMAL_IN)
|
||||
$(QUIET_AR)$(RM) $@ && $(AR) rcs $@ $(LIBTHERMAL_IN)
|
||||
|
||||
$(LIBTHERMAL_SO): $(LIBTHERMAL_IN)
|
||||
$(QUIET_LINK)$(CC) --shared -Wl,-soname,libthermal.so \
|
||||
-Wl,--version-script=$(VERSION_SCRIPT) $^ -o $@
|
||||
@ln -sf $(@F) $(OUTPUT)libthermal.so
|
||||
@ln -sf $(@F) $(OUTPUT)libthermal.so.$(LIBTHERMAL_VERSION)
|
||||
|
||||
|
||||
libs: $(THERMAL_UAPI) $(LIBTHERMAL_A) $(LIBTHERMAL_SO) $(LIBTHERMAL_PC)
|
||||
|
||||
all: fixdep
|
||||
$(Q)$(MAKE) libs
|
||||
|
||||
clean:
|
||||
$(call QUIET_CLEAN, libthermal) $(RM) $(LIBTHERMAL_A) \
|
||||
*.o *~ *.a *.so *.so.$(VERSION) *.so.$(LIBTHERMAL_VERSION) .*.d .*.cmd LIBTHERMAL-CFLAGS $(LIBTHERMAL_PC)
|
||||
|
||||
$(LIBTHERMAL_PC):
|
||||
$(QUIET_GEN)sed -e "s|@PREFIX@|$(prefix)|" \
|
||||
-e "s|@LIBDIR@|$(libdir_SQ)|" \
|
||||
-e "s|@VERSION@|$(VERSION)|" \
|
||||
< libthermal.pc.template > $@
|
||||
|
||||
define do_install_mkdir
|
||||
if [ ! -d '$(DESTDIR_SQ)$1' ]; then \
|
||||
$(INSTALL) -d -m 755 '$(DESTDIR_SQ)$1'; \
|
||||
fi
|
||||
endef
|
||||
|
||||
define do_install
|
||||
if [ ! -d '$(DESTDIR_SQ)$2' ]; then \
|
||||
$(INSTALL) -d -m 755 '$(DESTDIR_SQ)$2'; \
|
||||
fi; \
|
||||
$(INSTALL) $1 $(if $3,-m $3,) '$(DESTDIR_SQ)$2'
|
||||
endef
|
||||
|
||||
install_lib: libs
|
||||
$(call QUIET_INSTALL, $(LIBTHERMAL_ALL)) \
|
||||
$(call do_install_mkdir,$(libdir_SQ)); \
|
||||
cp -fpR $(LIBTHERMAL_ALL) $(DESTDIR)$(libdir_SQ)
|
||||
|
||||
install_headers:
|
||||
$(call QUIET_INSTALL, headers) \
|
||||
$(call do_install,include/thermal.h,$(prefix)/include/thermal,644); \
|
||||
|
||||
install_pkgconfig: $(LIBTHERMAL_PC)
|
||||
$(call QUIET_INSTALL, $(LIBTHERMAL_PC)) \
|
||||
$(call do_install,$(LIBTHERMAL_PC),$(libdir_SQ)/pkgconfig,644)
|
||||
|
||||
install_doc:
|
||||
$(Q)$(MAKE) -C Documentation install-man install-html install-examples
|
||||
|
||||
install: install_lib install_headers install_pkgconfig
|
||||
|
||||
FORCE:
|
||||
|
||||
.PHONY: all install clean FORCE
|
|
@ -0,0 +1,349 @@
|
|||
// SPDX-License-Identifier: LGPL-2.1+
|
||||
// Copyright (C) 2022, Linaro Ltd - Daniel Lezcano <daniel.lezcano@linaro.org>
|
||||
#define _GNU_SOURCE
|
||||
#include <errno.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#include <thermal.h>
|
||||
#include "thermal_nl.h"
|
||||
|
||||
static struct nla_policy thermal_genl_policy[THERMAL_GENL_ATTR_MAX + 1] = {
|
||||
/* Thermal zone */
|
||||
[THERMAL_GENL_ATTR_TZ] = { .type = NLA_NESTED },
|
||||
[THERMAL_GENL_ATTR_TZ_ID] = { .type = NLA_U32 },
|
||||
[THERMAL_GENL_ATTR_TZ_TEMP] = { .type = NLA_U32 },
|
||||
[THERMAL_GENL_ATTR_TZ_TRIP] = { .type = NLA_NESTED },
|
||||
[THERMAL_GENL_ATTR_TZ_TRIP_ID] = { .type = NLA_U32 },
|
||||
[THERMAL_GENL_ATTR_TZ_TRIP_TEMP] = { .type = NLA_U32 },
|
||||
[THERMAL_GENL_ATTR_TZ_TRIP_TYPE] = { .type = NLA_U32 },
|
||||
[THERMAL_GENL_ATTR_TZ_TRIP_HYST] = { .type = NLA_U32 },
|
||||
[THERMAL_GENL_ATTR_TZ_MODE] = { .type = NLA_U32 },
|
||||
[THERMAL_GENL_ATTR_TZ_CDEV_WEIGHT] = { .type = NLA_U32 },
|
||||
[THERMAL_GENL_ATTR_TZ_NAME] = { .type = NLA_STRING },
|
||||
|
||||
/* Governor(s) */
|
||||
[THERMAL_GENL_ATTR_TZ_GOV] = { .type = NLA_NESTED },
|
||||
[THERMAL_GENL_ATTR_TZ_GOV_NAME] = { .type = NLA_STRING },
|
||||
|
||||
/* Cooling devices */
|
||||
[THERMAL_GENL_ATTR_CDEV] = { .type = NLA_NESTED },
|
||||
[THERMAL_GENL_ATTR_CDEV_ID] = { .type = NLA_U32 },
|
||||
[THERMAL_GENL_ATTR_CDEV_CUR_STATE] = { .type = NLA_U32 },
|
||||
[THERMAL_GENL_ATTR_CDEV_MAX_STATE] = { .type = NLA_U32 },
|
||||
[THERMAL_GENL_ATTR_CDEV_NAME] = { .type = NLA_STRING },
|
||||
};
|
||||
|
||||
static int parse_tz_get(struct genl_info *info, struct thermal_zone **tz)
|
||||
{
|
||||
struct nlattr *attr;
|
||||
struct thermal_zone *__tz = NULL;
|
||||
size_t size = 0;
|
||||
int rem;
|
||||
|
||||
nla_for_each_nested(attr, info->attrs[THERMAL_GENL_ATTR_TZ], rem) {
|
||||
|
||||
if (nla_type(attr) == THERMAL_GENL_ATTR_TZ_ID) {
|
||||
|
||||
size++;
|
||||
|
||||
__tz = realloc(__tz, sizeof(*__tz) * (size + 2));
|
||||
if (!__tz)
|
||||
return THERMAL_ERROR;
|
||||
|
||||
__tz[size - 1].id = nla_get_u32(attr);
|
||||
}
|
||||
|
||||
|
||||
if (nla_type(attr) == THERMAL_GENL_ATTR_TZ_NAME)
|
||||
nla_strlcpy(__tz[size - 1].name, attr,
|
||||
THERMAL_NAME_LENGTH);
|
||||
}
|
||||
|
||||
if (__tz)
|
||||
__tz[size].id = -1;
|
||||
|
||||
*tz = __tz;
|
||||
|
||||
return THERMAL_SUCCESS;
|
||||
}
|
||||
|
||||
static int parse_cdev_get(struct genl_info *info, struct thermal_cdev **cdev)
|
||||
{
|
||||
struct nlattr *attr;
|
||||
struct thermal_cdev *__cdev = NULL;
|
||||
size_t size = 0;
|
||||
int rem;
|
||||
|
||||
nla_for_each_nested(attr, info->attrs[THERMAL_GENL_ATTR_CDEV], rem) {
|
||||
|
||||
if (nla_type(attr) == THERMAL_GENL_ATTR_CDEV_ID) {
|
||||
|
||||
size++;
|
||||
|
||||
__cdev = realloc(__cdev, sizeof(*__cdev) * (size + 2));
|
||||
if (!__cdev)
|
||||
return THERMAL_ERROR;
|
||||
|
||||
__cdev[size - 1].id = nla_get_u32(attr);
|
||||
}
|
||||
|
||||
if (nla_type(attr) == THERMAL_GENL_ATTR_CDEV_NAME) {
|
||||
nla_strlcpy(__cdev[size - 1].name, attr,
|
||||
THERMAL_NAME_LENGTH);
|
||||
}
|
||||
|
||||
if (nla_type(attr) == THERMAL_GENL_ATTR_CDEV_CUR_STATE)
|
||||
__cdev[size - 1].cur_state = nla_get_u32(attr);
|
||||
|
||||
if (nla_type(attr) == THERMAL_GENL_ATTR_CDEV_MAX_STATE)
|
||||
__cdev[size - 1].max_state = nla_get_u32(attr);
|
||||
}
|
||||
|
||||
if (__cdev)
|
||||
__cdev[size].id = -1;
|
||||
|
||||
*cdev = __cdev;
|
||||
|
||||
return THERMAL_SUCCESS;
|
||||
}
|
||||
|
||||
static int parse_tz_get_trip(struct genl_info *info, struct thermal_zone *tz)
|
||||
{
|
||||
struct nlattr *attr;
|
||||
struct thermal_trip *__tt = NULL;
|
||||
size_t size = 0;
|
||||
int rem;
|
||||
|
||||
nla_for_each_nested(attr, info->attrs[THERMAL_GENL_ATTR_TZ_TRIP], rem) {
|
||||
|
||||
if (nla_type(attr) == THERMAL_GENL_ATTR_TZ_TRIP_ID) {
|
||||
|
||||
size++;
|
||||
|
||||
__tt = realloc(__tt, sizeof(*__tt) * (size + 2));
|
||||
if (!__tt)
|
||||
return THERMAL_ERROR;
|
||||
|
||||
__tt[size - 1].id = nla_get_u32(attr);
|
||||
}
|
||||
|
||||
if (nla_type(attr) == THERMAL_GENL_ATTR_TZ_TRIP_TYPE)
|
||||
__tt[size - 1].type = nla_get_u32(attr);
|
||||
|
||||
if (nla_type(attr) == THERMAL_GENL_ATTR_TZ_TRIP_TEMP)
|
||||
__tt[size - 1].temp = nla_get_u32(attr);
|
||||
|
||||
if (nla_type(attr) == THERMAL_GENL_ATTR_TZ_TRIP_HYST)
|
||||
__tt[size - 1].hyst = nla_get_u32(attr);
|
||||
}
|
||||
|
||||
if (__tt)
|
||||
__tt[size].id = -1;
|
||||
|
||||
tz->trip = __tt;
|
||||
|
||||
return THERMAL_SUCCESS;
|
||||
}
|
||||
|
||||
static int parse_tz_get_temp(struct genl_info *info, struct thermal_zone *tz)
|
||||
{
|
||||
int id = -1;
|
||||
|
||||
if (info->attrs[THERMAL_GENL_ATTR_TZ_ID])
|
||||
id = nla_get_u32(info->attrs[THERMAL_GENL_ATTR_TZ_ID]);
|
||||
|
||||
if (tz->id != id)
|
||||
return THERMAL_ERROR;
|
||||
|
||||
if (info->attrs[THERMAL_GENL_ATTR_TZ_TEMP])
|
||||
tz->temp = nla_get_u32(info->attrs[THERMAL_GENL_ATTR_TZ_TEMP]);
|
||||
|
||||
return THERMAL_SUCCESS;
|
||||
}
|
||||
|
||||
static int parse_tz_get_gov(struct genl_info *info, struct thermal_zone *tz)
|
||||
{
|
||||
int id = -1;
|
||||
|
||||
if (info->attrs[THERMAL_GENL_ATTR_TZ_ID])
|
||||
id = nla_get_u32(info->attrs[THERMAL_GENL_ATTR_TZ_ID]);
|
||||
|
||||
if (tz->id != id)
|
||||
return THERMAL_ERROR;
|
||||
|
||||
if (info->attrs[THERMAL_GENL_ATTR_TZ_GOV_NAME]) {
|
||||
nla_strlcpy(tz->governor,
|
||||
info->attrs[THERMAL_GENL_ATTR_TZ_GOV_NAME],
|
||||
THERMAL_NAME_LENGTH);
|
||||
}
|
||||
|
||||
return THERMAL_SUCCESS;
|
||||
}
|
||||
|
||||
static int handle_netlink(struct nl_cache_ops *unused,
|
||||
struct genl_cmd *cmd,
|
||||
struct genl_info *info, void *arg)
|
||||
{
|
||||
int ret;
|
||||
|
||||
switch (cmd->c_id) {
|
||||
|
||||
case THERMAL_GENL_CMD_TZ_GET_ID:
|
||||
ret = parse_tz_get(info, arg);
|
||||
break;
|
||||
|
||||
case THERMAL_GENL_CMD_CDEV_GET:
|
||||
ret = parse_cdev_get(info, arg);
|
||||
break;
|
||||
|
||||
case THERMAL_GENL_CMD_TZ_GET_TEMP:
|
||||
ret = parse_tz_get_temp(info, arg);
|
||||
break;
|
||||
|
||||
case THERMAL_GENL_CMD_TZ_GET_TRIP:
|
||||
ret = parse_tz_get_trip(info, arg);
|
||||
break;
|
||||
|
||||
case THERMAL_GENL_CMD_TZ_GET_GOV:
|
||||
ret = parse_tz_get_gov(info, arg);
|
||||
break;
|
||||
|
||||
default:
|
||||
return THERMAL_ERROR;
|
||||
};
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static struct genl_cmd thermal_cmds[] = {
|
||||
{
|
||||
.c_id = THERMAL_GENL_CMD_TZ_GET_ID,
|
||||
.c_name = (char *)"List thermal zones",
|
||||
.c_msg_parser = handle_netlink,
|
||||
.c_maxattr = THERMAL_GENL_ATTR_MAX,
|
||||
.c_attr_policy = thermal_genl_policy,
|
||||
},
|
||||
{
|
||||
.c_id = THERMAL_GENL_CMD_TZ_GET_GOV,
|
||||
.c_name = (char *)"Get governor",
|
||||
.c_msg_parser = handle_netlink,
|
||||
.c_maxattr = THERMAL_GENL_ATTR_MAX,
|
||||
.c_attr_policy = thermal_genl_policy,
|
||||
},
|
||||
{
|
||||
.c_id = THERMAL_GENL_CMD_TZ_GET_TEMP,
|
||||
.c_name = (char *)"Get thermal zone temperature",
|
||||
.c_msg_parser = handle_netlink,
|
||||
.c_maxattr = THERMAL_GENL_ATTR_MAX,
|
||||
.c_attr_policy = thermal_genl_policy,
|
||||
},
|
||||
{
|
||||
.c_id = THERMAL_GENL_CMD_TZ_GET_TRIP,
|
||||
.c_name = (char *)"Get thermal zone trip points",
|
||||
.c_msg_parser = handle_netlink,
|
||||
.c_maxattr = THERMAL_GENL_ATTR_MAX,
|
||||
.c_attr_policy = thermal_genl_policy,
|
||||
},
|
||||
{
|
||||
.c_id = THERMAL_GENL_CMD_CDEV_GET,
|
||||
.c_name = (char *)"Get cooling devices",
|
||||
.c_msg_parser = handle_netlink,
|
||||
.c_maxattr = THERMAL_GENL_ATTR_MAX,
|
||||
.c_attr_policy = thermal_genl_policy,
|
||||
},
|
||||
};
|
||||
|
||||
static struct genl_ops thermal_cmd_ops = {
|
||||
.o_name = (char *)"thermal",
|
||||
.o_cmds = thermal_cmds,
|
||||
.o_ncmds = ARRAY_SIZE(thermal_cmds),
|
||||
};
|
||||
|
||||
static thermal_error_t thermal_genl_auto(struct thermal_handler *th, int id, int cmd,
|
||||
int flags, void *arg)
|
||||
{
|
||||
struct nl_msg *msg;
|
||||
void *hdr;
|
||||
|
||||
msg = nlmsg_alloc();
|
||||
if (!msg)
|
||||
return THERMAL_ERROR;
|
||||
|
||||
hdr = genlmsg_put(msg, NL_AUTO_PORT, NL_AUTO_SEQ, thermal_cmd_ops.o_id,
|
||||
0, flags, cmd, THERMAL_GENL_VERSION);
|
||||
if (!hdr)
|
||||
return THERMAL_ERROR;
|
||||
|
||||
if (id >= 0 && nla_put_u32(msg, THERMAL_GENL_ATTR_TZ_ID, id))
|
||||
return THERMAL_ERROR;
|
||||
|
||||
if (nl_send_msg(th->sk_cmd, th->cb_cmd, msg, genl_handle_msg, arg))
|
||||
return THERMAL_ERROR;
|
||||
|
||||
nlmsg_free(msg);
|
||||
|
||||
return THERMAL_SUCCESS;
|
||||
}
|
||||
|
||||
thermal_error_t thermal_cmd_get_tz(struct thermal_handler *th, struct thermal_zone **tz)
|
||||
{
|
||||
return thermal_genl_auto(th, -1, THERMAL_GENL_CMD_TZ_GET_ID,
|
||||
NLM_F_DUMP | NLM_F_ACK, tz);
|
||||
}
|
||||
|
||||
thermal_error_t thermal_cmd_get_cdev(struct thermal_handler *th, struct thermal_cdev **tc)
|
||||
{
|
||||
return thermal_genl_auto(th, -1, THERMAL_GENL_CMD_CDEV_GET,
|
||||
NLM_F_DUMP | NLM_F_ACK, tc);
|
||||
}
|
||||
|
||||
thermal_error_t thermal_cmd_get_trip(struct thermal_handler *th, struct thermal_zone *tz)
|
||||
{
|
||||
return thermal_genl_auto(th, tz->id, THERMAL_GENL_CMD_TZ_GET_TRIP,
|
||||
0, tz);
|
||||
}
|
||||
|
||||
thermal_error_t thermal_cmd_get_governor(struct thermal_handler *th, struct thermal_zone *tz)
|
||||
{
|
||||
return thermal_genl_auto(th, tz->id, THERMAL_GENL_CMD_TZ_GET_GOV, 0, tz);
|
||||
}
|
||||
|
||||
thermal_error_t thermal_cmd_get_temp(struct thermal_handler *th, struct thermal_zone *tz)
|
||||
{
|
||||
return thermal_genl_auto(th, tz->id, THERMAL_GENL_CMD_TZ_GET_TEMP, 0, tz);
|
||||
}
|
||||
|
||||
thermal_error_t thermal_cmd_exit(struct thermal_handler *th)
|
||||
{
|
||||
if (genl_unregister_family(&thermal_cmd_ops))
|
||||
return THERMAL_ERROR;
|
||||
|
||||
nl_thermal_disconnect(th->sk_cmd, th->cb_cmd);
|
||||
|
||||
return THERMAL_SUCCESS;
|
||||
}
|
||||
|
||||
thermal_error_t thermal_cmd_init(struct thermal_handler *th)
|
||||
{
|
||||
int ret;
|
||||
int family;
|
||||
|
||||
if (nl_thermal_connect(&th->sk_cmd, &th->cb_cmd))
|
||||
return THERMAL_ERROR;
|
||||
|
||||
ret = genl_register_family(&thermal_cmd_ops);
|
||||
if (ret)
|
||||
return THERMAL_ERROR;
|
||||
|
||||
ret = genl_ops_resolve(th->sk_cmd, &thermal_cmd_ops);
|
||||
if (ret)
|
||||
return THERMAL_ERROR;
|
||||
|
||||
family = genl_ctrl_resolve(th->sk_cmd, "nlctrl");
|
||||
if (family != GENL_ID_CTRL)
|
||||
return THERMAL_ERROR;
|
||||
|
||||
return THERMAL_SUCCESS;
|
||||
}
|
|
@ -0,0 +1,164 @@
|
|||
// SPDX-License-Identifier: LGPL-2.1+
|
||||
// Copyright (C) 2022, Linaro Ltd - Daniel Lezcano <daniel.lezcano@linaro.org>
|
||||
#include <linux/netlink.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <unistd.h>
|
||||
|
||||
|
||||
#include <thermal.h>
|
||||
#include "thermal_nl.h"
|
||||
|
||||
/*
|
||||
* Optimization: fill this array to tell which event we do want to pay
|
||||
* attention to. That happens at init time with the ops
|
||||
* structure. Each ops will enable the event and the general handler
|
||||
* will be able to discard the event if there is not ops associated
|
||||
* with it.
|
||||
*/
|
||||
static int enabled_ops[__THERMAL_GENL_EVENT_MAX];
|
||||
|
||||
static int handle_thermal_event(struct nl_msg *n, void *arg)
|
||||
{
|
||||
struct nlmsghdr *nlh = nlmsg_hdr(n);
|
||||
struct genlmsghdr *genlhdr = genlmsg_hdr(nlh);
|
||||
struct nlattr *attrs[THERMAL_GENL_ATTR_MAX + 1];
|
||||
struct thermal_handler_param *thp = arg;
|
||||
struct thermal_events_ops *ops = &thp->th->ops->events;
|
||||
|
||||
genlmsg_parse(nlh, 0, attrs, THERMAL_GENL_ATTR_MAX, NULL);
|
||||
|
||||
arg = thp->arg;
|
||||
|
||||
/*
|
||||
* This is an event we don't care of, bail out.
|
||||
*/
|
||||
if (!enabled_ops[genlhdr->cmd])
|
||||
return THERMAL_SUCCESS;
|
||||
|
||||
switch (genlhdr->cmd) {
|
||||
|
||||
case THERMAL_GENL_EVENT_TZ_CREATE:
|
||||
return ops->tz_create(nla_get_string(attrs[THERMAL_GENL_ATTR_TZ_NAME]),
|
||||
nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_ID]), arg);
|
||||
|
||||
case THERMAL_GENL_EVENT_TZ_DELETE:
|
||||
return ops->tz_delete(nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_ID]), arg);
|
||||
|
||||
case THERMAL_GENL_EVENT_TZ_ENABLE:
|
||||
return ops->tz_enable(nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_ID]), arg);
|
||||
|
||||
case THERMAL_GENL_EVENT_TZ_DISABLE:
|
||||
return ops->tz_disable(nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_ID]), arg);
|
||||
|
||||
case THERMAL_GENL_EVENT_TZ_TRIP_CHANGE:
|
||||
return ops->trip_change(nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_ID]),
|
||||
nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_TRIP_ID]),
|
||||
nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_TRIP_TYPE]),
|
||||
nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_TRIP_TEMP]),
|
||||
nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_TRIP_HYST]), arg);
|
||||
|
||||
case THERMAL_GENL_EVENT_TZ_TRIP_ADD:
|
||||
return ops->trip_add(nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_ID]),
|
||||
nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_TRIP_ID]),
|
||||
nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_TRIP_TYPE]),
|
||||
nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_TRIP_TEMP]),
|
||||
nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_TRIP_HYST]), arg);
|
||||
|
||||
case THERMAL_GENL_EVENT_TZ_TRIP_DELETE:
|
||||
return ops->trip_delete(nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_ID]),
|
||||
nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_TRIP_ID]), arg);
|
||||
|
||||
case THERMAL_GENL_EVENT_TZ_TRIP_UP:
|
||||
return ops->trip_high(nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_ID]),
|
||||
nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_TRIP_ID]),
|
||||
nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_TEMP]), arg);
|
||||
|
||||
case THERMAL_GENL_EVENT_TZ_TRIP_DOWN:
|
||||
return ops->trip_low(nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_ID]),
|
||||
nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_TRIP_ID]),
|
||||
nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_TEMP]), arg);
|
||||
|
||||
case THERMAL_GENL_EVENT_CDEV_ADD:
|
||||
return ops->cdev_add(nla_get_string(attrs[THERMAL_GENL_ATTR_CDEV_NAME]),
|
||||
nla_get_u32(attrs[THERMAL_GENL_ATTR_CDEV_ID]),
|
||||
nla_get_u32(attrs[THERMAL_GENL_ATTR_CDEV_MAX_STATE]), arg);
|
||||
|
||||
case THERMAL_GENL_EVENT_CDEV_DELETE:
|
||||
return ops->cdev_delete(nla_get_u32(attrs[THERMAL_GENL_ATTR_CDEV_ID]), arg);
|
||||
|
||||
case THERMAL_GENL_EVENT_CDEV_STATE_UPDATE:
|
||||
return ops->cdev_update(nla_get_u32(attrs[THERMAL_GENL_ATTR_CDEV_ID]),
|
||||
nla_get_u32(attrs[THERMAL_GENL_ATTR_CDEV_CUR_STATE]), arg);
|
||||
|
||||
case THERMAL_GENL_EVENT_TZ_GOV_CHANGE:
|
||||
return ops->gov_change(nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_ID]),
|
||||
nla_get_string(attrs[THERMAL_GENL_ATTR_GOV_NAME]), arg);
|
||||
default:
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
static void thermal_events_ops_init(struct thermal_events_ops *ops)
|
||||
{
|
||||
enabled_ops[THERMAL_GENL_EVENT_TZ_CREATE] = !!ops->tz_create;
|
||||
enabled_ops[THERMAL_GENL_EVENT_TZ_DELETE] = !!ops->tz_delete;
|
||||
enabled_ops[THERMAL_GENL_EVENT_TZ_DISABLE] = !!ops->tz_disable;
|
||||
enabled_ops[THERMAL_GENL_EVENT_TZ_ENABLE] = !!ops->tz_enable;
|
||||
enabled_ops[THERMAL_GENL_EVENT_TZ_TRIP_UP] = !!ops->trip_high;
|
||||
enabled_ops[THERMAL_GENL_EVENT_TZ_TRIP_DOWN] = !!ops->trip_low;
|
||||
enabled_ops[THERMAL_GENL_EVENT_TZ_TRIP_CHANGE] = !!ops->trip_change;
|
||||
enabled_ops[THERMAL_GENL_EVENT_TZ_TRIP_ADD] = !!ops->trip_add;
|
||||
enabled_ops[THERMAL_GENL_EVENT_TZ_TRIP_DELETE] = !!ops->trip_delete;
|
||||
enabled_ops[THERMAL_GENL_EVENT_CDEV_ADD] = !!ops->cdev_add;
|
||||
enabled_ops[THERMAL_GENL_EVENT_CDEV_DELETE] = !!ops->cdev_delete;
|
||||
enabled_ops[THERMAL_GENL_EVENT_CDEV_STATE_UPDATE] = !!ops->cdev_update;
|
||||
enabled_ops[THERMAL_GENL_EVENT_TZ_GOV_CHANGE] = !!ops->gov_change;
|
||||
}
|
||||
|
||||
thermal_error_t thermal_events_handle(struct thermal_handler *th, void *arg)
|
||||
{
|
||||
struct thermal_handler_param thp = { .th = th, .arg = arg };
|
||||
|
||||
if (!th)
|
||||
return THERMAL_ERROR;
|
||||
|
||||
if (nl_cb_set(th->cb_event, NL_CB_VALID, NL_CB_CUSTOM,
|
||||
handle_thermal_event, &thp))
|
||||
return THERMAL_ERROR;
|
||||
|
||||
return nl_recvmsgs(th->sk_event, th->cb_event);
|
||||
}
|
||||
|
||||
int thermal_events_fd(struct thermal_handler *th)
|
||||
{
|
||||
if (!th)
|
||||
return -1;
|
||||
|
||||
return nl_socket_get_fd(th->sk_event);
|
||||
}
|
||||
|
||||
thermal_error_t thermal_events_exit(struct thermal_handler *th)
|
||||
{
|
||||
if (nl_unsubscribe_thermal(th->sk_event, th->cb_event,
|
||||
THERMAL_GENL_EVENT_GROUP_NAME))
|
||||
return THERMAL_ERROR;
|
||||
|
||||
nl_thermal_disconnect(th->sk_event, th->cb_event);
|
||||
|
||||
return THERMAL_SUCCESS;
|
||||
}
|
||||
|
||||
thermal_error_t thermal_events_init(struct thermal_handler *th)
|
||||
{
|
||||
thermal_events_ops_init(&th->ops->events);
|
||||
|
||||
if (nl_thermal_connect(&th->sk_event, &th->cb_event))
|
||||
return THERMAL_ERROR;
|
||||
|
||||
if (nl_subscribe_thermal(th->sk_event, th->cb_event,
|
||||
THERMAL_GENL_EVENT_GROUP_NAME))
|
||||
return THERMAL_ERROR;
|
||||
|
||||
return THERMAL_SUCCESS;
|
||||
}
|
|
@ -0,0 +1,142 @@
|
|||
/* SPDX-License-Identifier: LGPL-2.1+ */
|
||||
/* Copyright (C) 2022, Linaro Ltd - Daniel Lezcano <daniel.lezcano@linaro.org> */
|
||||
#ifndef __LIBTHERMAL_H
|
||||
#define __LIBTHERMAL_H
|
||||
|
||||
#include <linux/thermal.h>
|
||||
|
||||
#ifndef LIBTHERMAL_API
|
||||
#define LIBTHERMAL_API __attribute__((visibility("default")))
|
||||
#endif
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
struct thermal_sampling_ops {
|
||||
int (*tz_temp)(int tz_id, int temp, void *arg);
|
||||
};
|
||||
|
||||
struct thermal_events_ops {
|
||||
int (*tz_create)(const char *name, int tz_id, void *arg);
|
||||
int (*tz_delete)(int tz_id, void *arg);
|
||||
int (*tz_enable)(int tz_id, void *arg);
|
||||
int (*tz_disable)(int tz_id, void *arg);
|
||||
int (*trip_high)(int tz_id, int trip_id, int temp, void *arg);
|
||||
int (*trip_low)(int tz_id, int trip_id, int temp, void *arg);
|
||||
int (*trip_add)(int tz_id, int trip_id, int type, int temp, int hyst, void *arg);
|
||||
int (*trip_change)(int tz_id, int trip_id, int type, int temp, int hyst, void *arg);
|
||||
int (*trip_delete)(int tz_id, int trip_id, void *arg);
|
||||
int (*cdev_add)(const char *name, int cdev_id, int max_state, void *arg);
|
||||
int (*cdev_delete)(int cdev_id, void *arg);
|
||||
int (*cdev_update)(int cdev_id, int cur_state, void *arg);
|
||||
int (*gov_change)(int tz_id, const char *gov_name, void *arg);
|
||||
};
|
||||
|
||||
struct thermal_ops {
|
||||
struct thermal_sampling_ops sampling;
|
||||
struct thermal_events_ops events;
|
||||
};
|
||||
|
||||
struct thermal_trip {
|
||||
int id;
|
||||
int type;
|
||||
int temp;
|
||||
int hyst;
|
||||
};
|
||||
|
||||
struct thermal_zone {
|
||||
int id;
|
||||
int temp;
|
||||
char name[THERMAL_NAME_LENGTH];
|
||||
char governor[THERMAL_NAME_LENGTH];
|
||||
struct thermal_trip *trip;
|
||||
};
|
||||
|
||||
struct thermal_cdev {
|
||||
int id;
|
||||
char name[THERMAL_NAME_LENGTH];
|
||||
int max_state;
|
||||
int min_state;
|
||||
int cur_state;
|
||||
};
|
||||
|
||||
typedef enum {
|
||||
THERMAL_ERROR = -1,
|
||||
THERMAL_SUCCESS = 0,
|
||||
} thermal_error_t;
|
||||
|
||||
struct thermal_handler;
|
||||
|
||||
typedef int (*cb_tz_t)(struct thermal_zone *, void *);
|
||||
|
||||
typedef int (*cb_tt_t)(struct thermal_trip *, void *);
|
||||
|
||||
typedef int (*cb_tc_t)(struct thermal_cdev *, void *);
|
||||
|
||||
LIBTHERMAL_API int for_each_thermal_zone(struct thermal_zone *tz, cb_tz_t cb, void *arg);
|
||||
|
||||
LIBTHERMAL_API int for_each_thermal_trip(struct thermal_trip *tt, cb_tt_t cb, void *arg);
|
||||
|
||||
LIBTHERMAL_API int for_each_thermal_cdev(struct thermal_cdev *cdev, cb_tc_t cb, void *arg);
|
||||
|
||||
LIBTHERMAL_API struct thermal_zone *thermal_zone_find_by_name(struct thermal_zone *tz,
|
||||
const char *name);
|
||||
|
||||
LIBTHERMAL_API struct thermal_zone *thermal_zone_find_by_id(struct thermal_zone *tz, int id);
|
||||
|
||||
LIBTHERMAL_API struct thermal_zone *thermal_zone_discover(struct thermal_handler *th);
|
||||
|
||||
LIBTHERMAL_API struct thermal_handler *thermal_init(struct thermal_ops *ops);
|
||||
|
||||
LIBTHERMAL_API void thermal_exit(struct thermal_handler *th);
|
||||
|
||||
/*
|
||||
* Netlink thermal events
|
||||
*/
|
||||
LIBTHERMAL_API thermal_error_t thermal_events_exit(struct thermal_handler *th);
|
||||
|
||||
LIBTHERMAL_API thermal_error_t thermal_events_init(struct thermal_handler *th);
|
||||
|
||||
LIBTHERMAL_API thermal_error_t thermal_events_handle(struct thermal_handler *th, void *arg);
|
||||
|
||||
LIBTHERMAL_API int thermal_events_fd(struct thermal_handler *th);
|
||||
|
||||
/*
|
||||
* Netlink thermal commands
|
||||
*/
|
||||
LIBTHERMAL_API thermal_error_t thermal_cmd_exit(struct thermal_handler *th);
|
||||
|
||||
LIBTHERMAL_API thermal_error_t thermal_cmd_init(struct thermal_handler *th);
|
||||
|
||||
LIBTHERMAL_API thermal_error_t thermal_cmd_get_tz(struct thermal_handler *th,
|
||||
struct thermal_zone **tz);
|
||||
|
||||
LIBTHERMAL_API thermal_error_t thermal_cmd_get_cdev(struct thermal_handler *th,
|
||||
struct thermal_cdev **tc);
|
||||
|
||||
LIBTHERMAL_API thermal_error_t thermal_cmd_get_trip(struct thermal_handler *th,
|
||||
struct thermal_zone *tz);
|
||||
|
||||
LIBTHERMAL_API thermal_error_t thermal_cmd_get_governor(struct thermal_handler *th,
|
||||
struct thermal_zone *tz);
|
||||
|
||||
LIBTHERMAL_API thermal_error_t thermal_cmd_get_temp(struct thermal_handler *th,
|
||||
struct thermal_zone *tz);
|
||||
|
||||
/*
|
||||
* Netlink thermal samples
|
||||
*/
|
||||
LIBTHERMAL_API thermal_error_t thermal_sampling_exit(struct thermal_handler *th);
|
||||
|
||||
LIBTHERMAL_API thermal_error_t thermal_sampling_init(struct thermal_handler *th);
|
||||
|
||||
LIBTHERMAL_API thermal_error_t thermal_sampling_handle(struct thermal_handler *th, void *arg);
|
||||
|
||||
LIBTHERMAL_API int thermal_sampling_fd(struct thermal_handler *th);
|
||||
|
||||
#endif /* __LIBTHERMAL_H */
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
|
@ -0,0 +1,25 @@
|
|||
LIBTHERMAL_0.0.1 {
|
||||
global:
|
||||
thermal_init;
|
||||
for_each_thermal_zone;
|
||||
for_each_thermal_trip;
|
||||
for_each_thermal_cdev;
|
||||
thermal_zone_find_by_name;
|
||||
thermal_zone_find_by_id;
|
||||
thermal_zone_discover;
|
||||
thermal_init;
|
||||
thermal_events_init;
|
||||
thermal_events_handle;
|
||||
thermal_events_fd;
|
||||
thermal_cmd_init;
|
||||
thermal_cmd_get_tz;
|
||||
thermal_cmd_get_cdev;
|
||||
thermal_cmd_get_trip;
|
||||
thermal_cmd_get_governor;
|
||||
thermal_cmd_get_temp;
|
||||
thermal_sampling_init;
|
||||
thermal_sampling_handle;
|
||||
thermal_sampling_fd;
|
||||
local:
|
||||
*;
|
||||
};
|
|
@ -0,0 +1,12 @@
|
|||
# SPDX-License-Identifier: (LGPL-2.1 OR BSD-2-Clause)
|
||||
|
||||
prefix=@PREFIX@
|
||||
libdir=@LIBDIR@
|
||||
includedir=${prefix}/include
|
||||
|
||||
Name: libthermal
|
||||
Description: thermal library
|
||||
Requires: libnl-3.0 libnl-genl-3.0
|
||||
Version: @VERSION@
|
||||
Libs: -L${libdir} -lnl-genl-3 -lnl-3
|
||||
Cflags: -I${includedir} -I{include}/libnl3
|
|
@ -0,0 +1,75 @@
|
|||
// SPDX-License-Identifier: LGPL-2.1+
|
||||
// Copyright (C) 2022, Linaro Ltd - Daniel Lezcano <daniel.lezcano@linaro.org>
|
||||
#include <errno.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#include <thermal.h>
|
||||
#include "thermal_nl.h"
|
||||
|
||||
static int handle_thermal_sample(struct nl_msg *n, void *arg)
|
||||
{
|
||||
struct nlmsghdr *nlh = nlmsg_hdr(n);
|
||||
struct genlmsghdr *genlhdr = genlmsg_hdr(nlh);
|
||||
struct nlattr *attrs[THERMAL_GENL_ATTR_MAX + 1];
|
||||
struct thermal_handler_param *thp = arg;
|
||||
struct thermal_handler *th = thp->th;
|
||||
|
||||
genlmsg_parse(nlh, 0, attrs, THERMAL_GENL_ATTR_MAX, NULL);
|
||||
|
||||
switch (genlhdr->cmd) {
|
||||
|
||||
case THERMAL_GENL_SAMPLING_TEMP:
|
||||
return th->ops->sampling.tz_temp(
|
||||
nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_ID]),
|
||||
nla_get_u32(attrs[THERMAL_GENL_ATTR_TZ_TEMP]), arg);
|
||||
default:
|
||||
return THERMAL_ERROR;
|
||||
}
|
||||
}
|
||||
|
||||
thermal_error_t thermal_sampling_handle(struct thermal_handler *th, void *arg)
|
||||
{
|
||||
struct thermal_handler_param thp = { .th = th, .arg = arg };
|
||||
|
||||
if (!th)
|
||||
return THERMAL_ERROR;
|
||||
|
||||
if (nl_cb_set(th->cb_sampling, NL_CB_VALID, NL_CB_CUSTOM,
|
||||
handle_thermal_sample, &thp))
|
||||
return THERMAL_ERROR;
|
||||
|
||||
return nl_recvmsgs(th->sk_sampling, th->cb_sampling);
|
||||
}
|
||||
|
||||
int thermal_sampling_fd(struct thermal_handler *th)
|
||||
{
|
||||
if (!th)
|
||||
return -1;
|
||||
|
||||
return nl_socket_get_fd(th->sk_sampling);
|
||||
}
|
||||
|
||||
thermal_error_t thermal_sampling_exit(struct thermal_handler *th)
|
||||
{
|
||||
if (nl_unsubscribe_thermal(th->sk_sampling, th->cb_sampling,
|
||||
THERMAL_GENL_EVENT_GROUP_NAME))
|
||||
return THERMAL_ERROR;
|
||||
|
||||
nl_thermal_disconnect(th->sk_sampling, th->cb_sampling);
|
||||
|
||||
return THERMAL_SUCCESS;
|
||||
}
|
||||
|
||||
thermal_error_t thermal_sampling_init(struct thermal_handler *th)
|
||||
{
|
||||
if (nl_thermal_connect(&th->sk_sampling, &th->cb_sampling))
|
||||
return THERMAL_ERROR;
|
||||
|
||||
if (nl_subscribe_thermal(th->sk_sampling, th->cb_sampling,
|
||||
THERMAL_GENL_SAMPLING_GROUP_NAME))
|
||||
return THERMAL_ERROR;
|
||||
|
||||
return THERMAL_SUCCESS;
|
||||
}
|
|
@ -0,0 +1,135 @@
|
|||
// SPDX-License-Identifier: LGPL-2.1+
|
||||
// Copyright (C) 2022, Linaro Ltd - Daniel Lezcano <daniel.lezcano@linaro.org>
|
||||
#include <stdio.h>
|
||||
#include <thermal.h>
|
||||
|
||||
#include "thermal_nl.h"
|
||||
|
||||
int for_each_thermal_cdev(struct thermal_cdev *cdev, cb_tc_t cb, void *arg)
|
||||
{
|
||||
int i, ret = 0;
|
||||
|
||||
if (!cdev)
|
||||
return 0;
|
||||
|
||||
for (i = 0; cdev[i].id != -1; i++)
|
||||
ret |= cb(&cdev[i], arg);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
int for_each_thermal_trip(struct thermal_trip *tt, cb_tt_t cb, void *arg)
|
||||
{
|
||||
int i, ret = 0;
|
||||
|
||||
if (!tt)
|
||||
return 0;
|
||||
|
||||
for (i = 0; tt[i].id != -1; i++)
|
||||
ret |= cb(&tt[i], arg);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
int for_each_thermal_zone(struct thermal_zone *tz, cb_tz_t cb, void *arg)
|
||||
{
|
||||
int i, ret = 0;
|
||||
|
||||
if (!tz)
|
||||
return 0;
|
||||
|
||||
for (i = 0; tz[i].id != -1; i++)
|
||||
ret |= cb(&tz[i], arg);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
struct thermal_zone *thermal_zone_find_by_name(struct thermal_zone *tz,
|
||||
const char *name)
|
||||
{
|
||||
int i;
|
||||
|
||||
if (!tz || !name)
|
||||
return NULL;
|
||||
|
||||
for (i = 0; tz[i].id != -1; i++) {
|
||||
if (!strcmp(tz[i].name, name))
|
||||
return &tz[i];
|
||||
}
|
||||
|
||||
return NULL;
|
||||
}
|
||||
|
||||
struct thermal_zone *thermal_zone_find_by_id(struct thermal_zone *tz, int id)
|
||||
{
|
||||
int i;
|
||||
|
||||
if (!tz || id < 0)
|
||||
return NULL;
|
||||
|
||||
for (i = 0; tz[i].id != -1; i++) {
|
||||
if (tz[i].id == id)
|
||||
return &tz[i];
|
||||
}
|
||||
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static int __thermal_zone_discover(struct thermal_zone *tz, void *th)
|
||||
{
|
||||
if (thermal_cmd_get_trip(th, tz) < 0)
|
||||
return -1;
|
||||
|
||||
if (thermal_cmd_get_governor(th, tz))
|
||||
return -1;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
struct thermal_zone *thermal_zone_discover(struct thermal_handler *th)
|
||||
{
|
||||
struct thermal_zone *tz;
|
||||
|
||||
if (thermal_cmd_get_tz(th, &tz) < 0)
|
||||
return NULL;
|
||||
|
||||
if (for_each_thermal_zone(tz, __thermal_zone_discover, th))
|
||||
return NULL;
|
||||
|
||||
return tz;
|
||||
}
|
||||
|
||||
void thermal_exit(struct thermal_handler *th)
|
||||
{
|
||||
thermal_cmd_exit(th);
|
||||
thermal_events_exit(th);
|
||||
thermal_sampling_exit(th);
|
||||
|
||||
free(th);
|
||||
}
|
||||
|
||||
struct thermal_handler *thermal_init(struct thermal_ops *ops)
|
||||
{
|
||||
struct thermal_handler *th;
|
||||
|
||||
th = malloc(sizeof(*th));
|
||||
if (!th)
|
||||
return NULL;
|
||||
th->ops = ops;
|
||||
|
||||
if (thermal_events_init(th))
|
||||
goto out_free;
|
||||
|
||||
if (thermal_sampling_init(th))
|
||||
goto out_free;
|
||||
|
||||
if (thermal_cmd_init(th))
|
||||
goto out_free;
|
||||
|
||||
return th;
|
||||
|
||||
out_free:
|
||||
free(th);
|
||||
|
||||
return NULL;
|
||||
}
|
|
@ -0,0 +1,215 @@
|
|||
// SPDX-License-Identifier: LGPL-2.1+
|
||||
// Copyright (C) 2022, Linaro Ltd - Daniel Lezcano <daniel.lezcano@linaro.org>
|
||||
#include <errno.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#include <thermal.h>
|
||||
#include "thermal_nl.h"
|
||||
|
||||
struct handler_args {
|
||||
const char *group;
|
||||
int id;
|
||||
};
|
||||
|
||||
static __thread int err;
|
||||
static __thread int done;
|
||||
|
||||
static int nl_seq_check_handler(struct nl_msg *msg, void *arg)
|
||||
{
|
||||
return NL_OK;
|
||||
}
|
||||
|
||||
static int nl_error_handler(struct sockaddr_nl *nla, struct nlmsgerr *nl_err,
|
||||
void *arg)
|
||||
{
|
||||
int *ret = arg;
|
||||
|
||||
if (ret)
|
||||
*ret = nl_err->error;
|
||||
|
||||
return NL_STOP;
|
||||
}
|
||||
|
||||
static int nl_finish_handler(struct nl_msg *msg, void *arg)
|
||||
{
|
||||
int *ret = arg;
|
||||
|
||||
if (ret)
|
||||
*ret = 1;
|
||||
|
||||
return NL_OK;
|
||||
}
|
||||
|
||||
static int nl_ack_handler(struct nl_msg *msg, void *arg)
|
||||
{
|
||||
int *ret = arg;
|
||||
|
||||
if (ret)
|
||||
*ret = 1;
|
||||
|
||||
return NL_OK;
|
||||
}
|
||||
|
||||
int nl_send_msg(struct nl_sock *sock, struct nl_cb *cb, struct nl_msg *msg,
|
||||
int (*rx_handler)(struct nl_msg *, void *), void *data)
|
||||
{
|
||||
if (!rx_handler)
|
||||
return THERMAL_ERROR;
|
||||
|
||||
err = nl_send_auto_complete(sock, msg);
|
||||
if (err < 0)
|
||||
return err;
|
||||
|
||||
nl_cb_set(cb, NL_CB_VALID, NL_CB_CUSTOM, rx_handler, data);
|
||||
|
||||
err = done = 0;
|
||||
|
||||
while (err == 0 && done == 0)
|
||||
nl_recvmsgs(sock, cb);
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
static int nl_family_handler(struct nl_msg *msg, void *arg)
|
||||
{
|
||||
struct handler_args *grp = arg;
|
||||
struct nlattr *tb[CTRL_ATTR_MAX + 1];
|
||||
struct genlmsghdr *gnlh = nlmsg_data(nlmsg_hdr(msg));
|
||||
struct nlattr *mcgrp;
|
||||
int rem_mcgrp;
|
||||
|
||||
nla_parse(tb, CTRL_ATTR_MAX, genlmsg_attrdata(gnlh, 0),
|
||||
genlmsg_attrlen(gnlh, 0), NULL);
|
||||
|
||||
if (!tb[CTRL_ATTR_MCAST_GROUPS])
|
||||
return THERMAL_ERROR;
|
||||
|
||||
nla_for_each_nested(mcgrp, tb[CTRL_ATTR_MCAST_GROUPS], rem_mcgrp) {
|
||||
|
||||
struct nlattr *tb_mcgrp[CTRL_ATTR_MCAST_GRP_MAX + 1];
|
||||
|
||||
nla_parse(tb_mcgrp, CTRL_ATTR_MCAST_GRP_MAX,
|
||||
nla_data(mcgrp), nla_len(mcgrp), NULL);
|
||||
|
||||
if (!tb_mcgrp[CTRL_ATTR_MCAST_GRP_NAME] ||
|
||||
!tb_mcgrp[CTRL_ATTR_MCAST_GRP_ID])
|
||||
continue;
|
||||
|
||||
if (strncmp(nla_data(tb_mcgrp[CTRL_ATTR_MCAST_GRP_NAME]),
|
||||
grp->group,
|
||||
nla_len(tb_mcgrp[CTRL_ATTR_MCAST_GRP_NAME])))
|
||||
continue;
|
||||
|
||||
grp->id = nla_get_u32(tb_mcgrp[CTRL_ATTR_MCAST_GRP_ID]);
|
||||
|
||||
break;
|
||||
}
|
||||
|
||||
return THERMAL_SUCCESS;
|
||||
}
|
||||
|
||||
static int nl_get_multicast_id(struct nl_sock *sock, struct nl_cb *cb,
|
||||
const char *family, const char *group)
|
||||
{
|
||||
struct nl_msg *msg;
|
||||
int ret = 0, ctrlid;
|
||||
struct handler_args grp = {
|
||||
.group = group,
|
||||
.id = -ENOENT,
|
||||
};
|
||||
|
||||
msg = nlmsg_alloc();
|
||||
if (!msg)
|
||||
return THERMAL_ERROR;
|
||||
|
||||
ctrlid = genl_ctrl_resolve(sock, "nlctrl");
|
||||
|
||||
genlmsg_put(msg, 0, 0, ctrlid, 0, 0, CTRL_CMD_GETFAMILY, 0);
|
||||
|
||||
nla_put_string(msg, CTRL_ATTR_FAMILY_NAME, family);
|
||||
|
||||
ret = nl_send_msg(sock, cb, msg, nl_family_handler, &grp);
|
||||
if (ret)
|
||||
goto nla_put_failure;
|
||||
|
||||
ret = grp.id;
|
||||
|
||||
nla_put_failure:
|
||||
nlmsg_free(msg);
|
||||
return ret;
|
||||
}
|
||||
|
||||
int nl_thermal_connect(struct nl_sock **nl_sock, struct nl_cb **nl_cb)
|
||||
{
|
||||
struct nl_cb *cb;
|
||||
struct nl_sock *sock;
|
||||
|
||||
cb = nl_cb_alloc(NL_CB_DEFAULT);
|
||||
if (!cb)
|
||||
return THERMAL_ERROR;
|
||||
|
||||
sock = nl_socket_alloc();
|
||||
if (!sock)
|
||||
goto out_cb_free;
|
||||
|
||||
if (genl_connect(sock))
|
||||
goto out_socket_free;
|
||||
|
||||
if (nl_cb_err(cb, NL_CB_CUSTOM, nl_error_handler, &err) ||
|
||||
nl_cb_set(cb, NL_CB_FINISH, NL_CB_CUSTOM, nl_finish_handler, &done) ||
|
||||
nl_cb_set(cb, NL_CB_ACK, NL_CB_CUSTOM, nl_ack_handler, &done) ||
|
||||
nl_cb_set(cb, NL_CB_SEQ_CHECK, NL_CB_CUSTOM, nl_seq_check_handler, &done))
|
||||
return THERMAL_ERROR;
|
||||
|
||||
*nl_sock = sock;
|
||||
*nl_cb = cb;
|
||||
|
||||
return THERMAL_SUCCESS;
|
||||
|
||||
out_socket_free:
|
||||
nl_socket_free(sock);
|
||||
out_cb_free:
|
||||
nl_cb_put(cb);
|
||||
return THERMAL_ERROR;
|
||||
}
|
||||
|
||||
void nl_thermal_disconnect(struct nl_sock *nl_sock, struct nl_cb *nl_cb)
|
||||
{
|
||||
nl_close(nl_sock);
|
||||
nl_socket_free(nl_sock);
|
||||
nl_cb_put(nl_cb);
|
||||
}
|
||||
|
||||
int nl_unsubscribe_thermal(struct nl_sock *nl_sock, struct nl_cb *nl_cb,
|
||||
const char *group)
|
||||
{
|
||||
int mcid;
|
||||
|
||||
mcid = nl_get_multicast_id(nl_sock, nl_cb, THERMAL_GENL_FAMILY_NAME,
|
||||
group);
|
||||
if (mcid < 0)
|
||||
return THERMAL_ERROR;
|
||||
|
||||
if (nl_socket_drop_membership(nl_sock, mcid))
|
||||
return THERMAL_ERROR;
|
||||
|
||||
return THERMAL_SUCCESS;
|
||||
}
|
||||
|
||||
int nl_subscribe_thermal(struct nl_sock *nl_sock, struct nl_cb *nl_cb,
|
||||
const char *group)
|
||||
{
|
||||
int mcid;
|
||||
|
||||
mcid = nl_get_multicast_id(nl_sock, nl_cb, THERMAL_GENL_FAMILY_NAME,
|
||||
group);
|
||||
if (mcid < 0)
|
||||
return THERMAL_ERROR;
|
||||
|
||||
if (nl_socket_add_membership(nl_sock, mcid))
|
||||
return THERMAL_ERROR;
|
||||
|
||||
return THERMAL_SUCCESS;
|
||||
}
|
|
@ -0,0 +1,46 @@
|
|||
/* SPDX-License-Identifier: LGPL-2.1+ */
|
||||
/* Copyright (C) 2022, Linaro Ltd - Daniel Lezcano <daniel.lezcano@linaro.org> */
|
||||
#ifndef __THERMAL_H
|
||||
#define __THERMAL_H
|
||||
|
||||
#include <netlink/netlink.h>
|
||||
#include <netlink/genl/genl.h>
|
||||
#include <netlink/genl/mngt.h>
|
||||
#include <netlink/genl/ctrl.h>
|
||||
|
||||
struct thermal_handler {
|
||||
int done;
|
||||
int error;
|
||||
struct thermal_ops *ops;
|
||||
struct nl_msg *msg;
|
||||
struct nl_sock *sk_event;
|
||||
struct nl_sock *sk_sampling;
|
||||
struct nl_sock *sk_cmd;
|
||||
struct nl_cb *cb_cmd;
|
||||
struct nl_cb *cb_event;
|
||||
struct nl_cb *cb_sampling;
|
||||
};
|
||||
|
||||
struct thermal_handler_param {
|
||||
struct thermal_handler *th;
|
||||
void *arg;
|
||||
};
|
||||
|
||||
/*
|
||||
* Low level netlink
|
||||
*/
|
||||
extern int nl_subscribe_thermal(struct nl_sock *nl_sock, struct nl_cb *nl_cb,
|
||||
const char *group);
|
||||
|
||||
extern int nl_unsubscribe_thermal(struct nl_sock *nl_sock, struct nl_cb *nl_cb,
|
||||
const char *group);
|
||||
|
||||
extern int nl_thermal_connect(struct nl_sock **nl_sock, struct nl_cb **nl_cb);
|
||||
|
||||
extern void nl_thermal_disconnect(struct nl_sock *nl_sock, struct nl_cb *nl_cb);
|
||||
|
||||
extern int nl_send_msg(struct nl_sock *sock, struct nl_cb *nl_cb, struct nl_msg *msg,
|
||||
int (*rx_handler)(struct nl_msg *, void *),
|
||||
void *data);
|
||||
|
||||
#endif /* __THERMAL_H */
|
Loading…
Reference in New Issue