Thermal: Support cpufreq cooling device
Signed-off-by: wangjianyu3 <wangjianyu3@xiaomi.com>
This commit is contained in:
parent
338b5d7415
commit
2f285a04f9
@ -23,4 +23,11 @@ config THERMAL_GOVERNOR_STEP_WISE
|
|||||||
---help---
|
---help---
|
||||||
Enable step wise governor.
|
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
|
endif # THERMAL
|
||||||
|
@ -28,6 +28,10 @@ ifeq ($(CONFIG_THERMAL_GOVERNOR_STEP_WISE),y)
|
|||||||
CSRCS += thermal_step_wise.c
|
CSRCS += thermal_step_wise.c
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
ifeq ($(CONFIG_THERMAL_CDEV_CPUFREQ),y)
|
||||||
|
CSRCS += thermal_cpufreq_cooling.c
|
||||||
|
endif
|
||||||
|
|
||||||
DEPPATH += --dep-path thermal
|
DEPPATH += --dep-path thermal
|
||||||
VPATH += thermal
|
VPATH += thermal
|
||||||
|
|
||||||
|
@ -848,5 +848,12 @@ int thermal_init(void)
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef CONFIG_THERMAL_CDEV_CPUFREQ
|
||||||
|
if (NULL == thermal_cpufreq_cooling_register())
|
||||||
|
{
|
||||||
|
return -ENOTSUP;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -57,6 +57,11 @@ struct thermal_instance_s
|
|||||||
|
|
||||||
void
|
void
|
||||||
thermal_cooling_device_update (FAR struct thermal_cooling_device_s *cdev);
|
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 */
|
/* 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 New Issue
Block a user