Spdk/lib/event/dpdk_governor.c

163 lines
4.1 KiB
C
Raw Normal View History

/*-
* BSD LICENSE
*
* Copyright (c) Intel Corporation.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Intel Corporation nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "spdk/stdinc.h"
#include "spdk/log.h"
#include "spdk/event.h"
#include "spdk_internal/event.h"
#include <rte_power.h>
static uint32_t
_get_core_curr_freq(uint32_t lcore_id)
{
const uint32_t MAX_CORE_FREQ_NUM = 64;
uint32_t freqs[MAX_CORE_FREQ_NUM];
uint32_t freq_index;
int rc;
rc = rte_power_freqs(lcore_id, freqs, MAX_CORE_FREQ_NUM);
if (!rc) {
SPDK_ERRLOG("Unable to get current core frequency array for core %d\n.", lcore_id);
return 0;
}
freq_index = rte_power_get_freq(lcore_id);
if (freq_index >= MAX_CORE_FREQ_NUM) {
SPDK_ERRLOG("Unable to get current core frequency for core %d\n.", lcore_id);
return 0;
}
return freqs[freq_index];
}
static int
_core_freq_up(uint32_t lcore_id)
{
return rte_power_freq_up(lcore_id);
}
static int
_core_freq_down(uint32_t lcore_id)
{
return rte_power_freq_down(lcore_id);
}
static int
_set_core_freq_max(uint32_t lcore_id)
{
return rte_power_freq_max(lcore_id);
}
static int
_set_core_freq_min(uint32_t lcore_id)
{
return rte_power_freq_min(lcore_id);
}
static int
_get_core_capabilities(uint32_t lcore_id, struct spdk_governor_capabilities *capabilities)
{
struct rte_power_core_capabilities caps;
int rc;
rc = rte_power_get_capabilities(lcore_id, &caps);
if (rc != 0) {
return rc;
}
capabilities->priority = caps.priority == 0 ? false : true;
return 0;
}
static int
_init_core(uint32_t lcore_id)
{
struct rte_power_core_capabilities caps;
int rc;
rc = rte_power_init(lcore_id);
if (rc != 0) {
SPDK_ERRLOG("Failed to initialize on core%d\n", lcore_id);
return rc;
}
rc = rte_power_get_capabilities(lcore_id, &caps);
if (rc != 0) {
SPDK_ERRLOG("Failed retrievie capabilities of core%d\n", lcore_id);
return rc;
}
if (caps.turbo) {
rc = rte_power_freq_enable_turbo(lcore_id);
if (rc != 0) {
SPDK_ERRLOG("Failed to set turbo on core%d\n", lcore_id);
return rc;
}
}
return rc;
}
static int
_deinit_core(uint32_t lcore_id)
{
int rc;
rc = rte_power_exit(lcore_id);
if (rc) {
SPDK_ERRLOG("DPDK Power management library deinitialization failed on core%d\n", lcore_id);
}
return rc;
}
static struct spdk_governor dpdk_governor = {
.name = "dpdk_governor",
.get_core_curr_freq = _get_core_curr_freq,
.core_freq_up = _core_freq_up,
.core_freq_down = _core_freq_down,
.set_core_freq_max = _set_core_freq_max,
.set_core_freq_min = _set_core_freq_min,
.get_core_capabilities = _get_core_capabilities,
.init_core = _init_core,
.deinit_core = _deinit_core,
.init = NULL,
.deinit = NULL,
};
SPDK_GOVERNOR_REGISTER(&dpdk_governor);