Part of #2256 * calculated * changing * deferred * deinitialize * initialization * particular * receive * request * retrieve * satisfied * succeed * thread * unplugged * unregister Change-Id: I13e38f9160cb1a15a87cb5974785a34604124fa3 Signed-off-by: Josh Soref <jsoref@gmail.com> Reviewed-on: https://review.spdk.io/gerrit/c/spdk/spdk/+/10406 Community-CI: Mellanox Build Bot Community-CI: Broadcom CI <spdk-ci.pdl@broadcom.com> Tested-by: SPDK CI Jenkins <sys_sgci@intel.com> Reviewed-by: Shuhei Matsumoto <shuhei.matsumoto.xt@hitachi.com> Reviewed-by: Tomasz Zawadzki <tomasz.zawadzki@intel.com>
192 lines
4.5 KiB
C
192 lines
4.5 KiB
C
/*-
|
|
* 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/env.h"
|
|
#include "spdk/event.h"
|
|
#include "spdk/scheduler.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 retrieve 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
|
|
_init(void)
|
|
{
|
|
uint32_t i, j;
|
|
int rc = 0;
|
|
|
|
SPDK_ENV_FOREACH_CORE(i) {
|
|
rc = _init_core(i);
|
|
if (rc != 0) {
|
|
SPDK_ERRLOG("Failed to initialize on core%d\n", i);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (rc == 0) {
|
|
return rc;
|
|
}
|
|
|
|
/* When initialization of a core failed, deinitialize prior cores. */
|
|
SPDK_ENV_FOREACH_CORE(j) {
|
|
if (j >= i) {
|
|
break;
|
|
}
|
|
if (rte_power_exit(j) != 0) {
|
|
SPDK_ERRLOG("Failed to deinitialize on core%d\n", j);
|
|
}
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
static void
|
|
_deinit(void)
|
|
{
|
|
uint32_t i;
|
|
|
|
SPDK_ENV_FOREACH_CORE(i) {
|
|
if (rte_power_exit(i) != 0) {
|
|
SPDK_ERRLOG("Failed to deinitialize on core%d\n", i);
|
|
}
|
|
}
|
|
}
|
|
|
|
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 = _init,
|
|
.deinit = _deinit,
|
|
};
|
|
|
|
SPDK_GOVERNOR_REGISTER(dpdk_governor);
|