mirror of
https://github.com/apache/nuttx.git
synced 2025-01-13 09:49:21 +08:00
Thermal: Support cpufreq cooling device
Signed-off-by: wangjianyu3 <wangjianyu3@xiaomi.com>
This commit is contained in:
parent
338b5d7415
commit
2f285a04f9
5 changed files with 257 additions and 0 deletions
|
@ -23,4 +23,11 @@ config THERMAL_GOVERNOR_STEP_WISE
|
|||
---help---
|
||||
Enable step wise governor.
|
||||
|
||||
config THERMAL_CDEV_CPUFREQ
|
||||
bool "Thermal cpufreq cooling device"
|
||||
default n
|
||||
depends on CPUFREQ
|
||||
---help---
|
||||
Enable thermal cpufreq cooling device.
|
||||
|
||||
endif # THERMAL
|
||||
|
|
|
@ -28,6 +28,10 @@ ifeq ($(CONFIG_THERMAL_GOVERNOR_STEP_WISE),y)
|
|||
CSRCS += thermal_step_wise.c
|
||||
endif
|
||||
|
||||
ifeq ($(CONFIG_THERMAL_CDEV_CPUFREQ),y)
|
||||
CSRCS += thermal_cpufreq_cooling.c
|
||||
endif
|
||||
|
||||
DEPPATH += --dep-path thermal
|
||||
VPATH += thermal
|
||||
|
||||
|
|
|
@ -848,5 +848,12 @@ int thermal_init(void)
|
|||
}
|
||||
#endif
|
||||
|
||||
#ifdef CONFIG_THERMAL_CDEV_CPUFREQ
|
||||
if (NULL == thermal_cpufreq_cooling_register())
|
||||
{
|
||||
return -ENOTSUP;
|
||||
}
|
||||
#endif
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
|
|
@ -57,6 +57,11 @@ struct thermal_instance_s
|
|||
|
||||
void
|
||||
thermal_cooling_device_update (FAR struct thermal_cooling_device_s *cdev);
|
||||
#ifdef CONFIG_THERMAL_CDEV_CPUFREQ
|
||||
FAR struct thermal_cooling_device_s *thermal_cpufreq_cooling_register(void);
|
||||
void thermal_cpufreq_cooling_unregister(
|
||||
FAR struct thermal_cooling_device_s *cdev);
|
||||
#endif /* CONFIG_THERMAL_CDEV_CPUFREQ */
|
||||
|
||||
/* Zone Device */
|
||||
|
||||
|
|
234
drivers/thermal/thermal_cpufreq_cooling.c
Normal file
234
drivers/thermal/thermal_cpufreq_cooling.c
Normal file
|
@ -0,0 +1,234 @@
|
|||
/****************************************************************************
|
||||
* drivers/thermal/thermal_cpufreq_cooling.c
|
||||
*
|
||||
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||
* contributor license agreements. See the NOTICE file distributed with
|
||||
* this work for additional information regarding copyright ownership. The
|
||||
* ASF licenses this file to you under the Apache License, Version 2.0 (the
|
||||
* "License"); you may not use this file except in compliance with the
|
||||
* License. You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||
* License for the specific language governing permissions and limitations
|
||||
* under the License.
|
||||
*
|
||||
****************************************************************************/
|
||||
|
||||
/****************************************************************************
|
||||
* Included Files
|
||||
****************************************************************************/
|
||||
|
||||
#include <nuttx/cpufreq.h>
|
||||
#include <nuttx/kmalloc.h>
|
||||
|
||||
#include "thermal_core.h"
|
||||
|
||||
/****************************************************************************
|
||||
* Private Types
|
||||
****************************************************************************/
|
||||
|
||||
struct cpufreq_cooling_device_s
|
||||
{
|
||||
FAR const struct cpufreq_frequency_table *table;
|
||||
FAR struct cpufreq_policy *policy;
|
||||
FAR struct cpufreq_qos *qos;
|
||||
unsigned int cur_state;
|
||||
unsigned int max_state;
|
||||
};
|
||||
|
||||
/****************************************************************************
|
||||
* Private Function Prototypes
|
||||
****************************************************************************/
|
||||
|
||||
static int cpufreq_get_max_state(FAR struct thermal_cooling_device_s *cdev,
|
||||
FAR unsigned int *state);
|
||||
static int cpufreq_get_state (FAR struct thermal_cooling_device_s *cdev,
|
||||
FAR unsigned int *state);
|
||||
static int cpufreq_set_state (FAR struct thermal_cooling_device_s *cdev,
|
||||
unsigned int state);
|
||||
|
||||
/****************************************************************************
|
||||
* Private Data
|
||||
****************************************************************************/
|
||||
|
||||
static const struct thermal_cooling_device_ops_s g_cpufreq_cdev_ops =
|
||||
{
|
||||
.set_state = cpufreq_set_state,
|
||||
.get_state = cpufreq_get_state,
|
||||
.get_max_state = cpufreq_get_max_state,
|
||||
};
|
||||
|
||||
/****************************************************************************
|
||||
* Private Functions
|
||||
****************************************************************************/
|
||||
|
||||
static int cpufreq_get_max_state(FAR struct thermal_cooling_device_s *cdev,
|
||||
FAR unsigned int *state)
|
||||
{
|
||||
struct cpufreq_cooling_device_s *cpufreq_cdev = cdev->devdata;
|
||||
|
||||
*state = cpufreq_cdev->max_state;
|
||||
return OK;
|
||||
}
|
||||
|
||||
static int cpufreq_get_state(FAR struct thermal_cooling_device_s *cdev,
|
||||
FAR unsigned int *state)
|
||||
{
|
||||
struct cpufreq_cooling_device_s *cpufreq_cdev = cdev->devdata;
|
||||
|
||||
*state = cpufreq_cdev->cur_state;
|
||||
return OK;
|
||||
}
|
||||
|
||||
static int cpufreq_set_state(FAR struct thermal_cooling_device_s *cdev,
|
||||
unsigned int state)
|
||||
{
|
||||
struct cpufreq_cooling_device_s *cpufreq_cdev = cdev->devdata;
|
||||
unsigned int index = cpufreq_cdev->max_state - state;
|
||||
int ret;
|
||||
|
||||
thinfo("CPU Freq cooling %u %u \n",
|
||||
cpufreq_cdev->table[index].frequency,
|
||||
cpufreq_cdev->table[index + 1].frequency);
|
||||
|
||||
if (cpufreq_cdev->qos == NULL)
|
||||
{
|
||||
cpufreq_cdev->qos = cpufreq_qos_add_request(
|
||||
cpufreq_cdev->policy,
|
||||
cpufreq_cdev->table[index].frequency,
|
||||
cpufreq_cdev->table[index + 1].frequency);
|
||||
if (!cpufreq_cdev->qos)
|
||||
{
|
||||
therr("Add qos request failed!");
|
||||
return -EINVAL;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
ret = cpufreq_qos_update_request(
|
||||
cpufreq_cdev->qos,
|
||||
cpufreq_cdev->table[index].frequency,
|
||||
cpufreq_cdev->table[index + 1].frequency);
|
||||
if (ret < 0)
|
||||
{
|
||||
therr("Update qos request failed!");
|
||||
return ret;
|
||||
}
|
||||
}
|
||||
|
||||
cpufreq_cdev->cur_state = state;
|
||||
return OK;
|
||||
}
|
||||
|
||||
/****************************************************************************
|
||||
* Public Functions
|
||||
****************************************************************************/
|
||||
|
||||
/****************************************************************************
|
||||
* Name: thermal_cpufreq_cooling_register
|
||||
*
|
||||
* Description:
|
||||
* Register cpufreq cooling device
|
||||
*
|
||||
* Input Parameters:
|
||||
* policy - cpufreq policy
|
||||
*
|
||||
* Returned Value:
|
||||
* Addr of created cooling device entry
|
||||
****************************************************************************/
|
||||
|
||||
FAR struct thermal_cooling_device_s *thermal_cpufreq_cooling_register(void)
|
||||
{
|
||||
FAR struct cpufreq_cooling_device_s *cpufreq_cdev;
|
||||
FAR const struct cpufreq_frequency_table *table;
|
||||
FAR struct thermal_cooling_device_s *cdev;
|
||||
FAR struct cpufreq_driver **driver;
|
||||
FAR struct cpufreq_policy *policy;
|
||||
unsigned int count;
|
||||
|
||||
policy = cpufreq_policy_get();
|
||||
if (policy == NULL)
|
||||
{
|
||||
therr("Get cpufreq policy failed!\n");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
driver = (FAR struct cpufreq_driver **)policy;
|
||||
|
||||
table = (*driver)->get_table(policy);
|
||||
if (table == NULL)
|
||||
{
|
||||
therr("Get cpufreq table failed!\n");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
for (count = 0; table[count].frequency != CPUFREQ_TABLE_END; count++)
|
||||
{
|
||||
}
|
||||
|
||||
if (count < 2)
|
||||
{
|
||||
therr("Invalid cpufreq table!\n");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
cpufreq_cdev = kmm_zalloc(sizeof(*cpufreq_cdev));
|
||||
if (cpufreq_cdev == NULL)
|
||||
{
|
||||
therr("No memory for cpufreq cooling device registering!\n");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
cpufreq_cdev->table = table;
|
||||
cpufreq_cdev->policy = policy;
|
||||
cpufreq_cdev->max_state = count - 2;
|
||||
thinfo("max level of cpufreq is %d \n", cpufreq_cdev->max_state);
|
||||
|
||||
cdev = thermal_cooling_device_register("cpufreq", cpufreq_cdev,
|
||||
&g_cpufreq_cdev_ops);
|
||||
if (cdev == NULL)
|
||||
{
|
||||
kmm_free(cpufreq_cdev);
|
||||
}
|
||||
|
||||
return cdev;
|
||||
}
|
||||
|
||||
/****************************************************************************
|
||||
* Name: thermal_cpufreq_cooling_unregister
|
||||
*
|
||||
* Description:
|
||||
* Unregister cpufreq cooling device
|
||||
*
|
||||
* Input Parameters:
|
||||
* cdev - Addr of cpufre cooling devcie entry
|
||||
*
|
||||
* Returned Value:
|
||||
* None
|
||||
****************************************************************************/
|
||||
|
||||
void
|
||||
thermal_cpufreq_cooling_unregister(FAR struct thermal_cooling_device_s *cdev)
|
||||
{
|
||||
struct cpufreq_cooling_device_s *cpufreq_cdev;
|
||||
int ret;
|
||||
|
||||
cpufreq_cdev = cdev->devdata;
|
||||
|
||||
if (cpufreq_cdev->qos)
|
||||
{
|
||||
ret = cpufreq_qos_remove_request(cpufreq_cdev->qos);
|
||||
if (ret < 0)
|
||||
{
|
||||
thinfo("ret=%d\n", ret);
|
||||
therr("Remove cpufreq qos failed!\n");
|
||||
}
|
||||
}
|
||||
|
||||
thermal_cooling_device_unregister(cdev);
|
||||
kmm_free(cpufreq_cdev);
|
||||
}
|
Loading…
Reference in a new issue