f-stack/dpdk/drivers/event/octeontx/timvf_evdev.c

453 lines
12 KiB
C

/* SPDX-License-Identifier: BSD-3-Clause
* Copyright(c) 2017 Cavium, Inc
*/
#include "ssovf_evdev.h"
#include "timvf_evdev.h"
RTE_LOG_REGISTER(otx_logtype_timvf, pmd.event.octeontx.timer, NOTICE);
static struct rte_eventdev *event_dev;
struct __rte_packed timvf_mbox_dev_info {
uint64_t ring_active[4];
uint64_t clk_freq;
};
/* Response messages */
enum {
MBOX_RET_SUCCESS,
MBOX_RET_INVALID,
MBOX_RET_INTERNAL_ERR,
};
static int
timvf_mbox_dev_info_get(struct timvf_mbox_dev_info *info)
{
struct octeontx_mbox_hdr hdr = {0};
uint16_t len = sizeof(struct timvf_mbox_dev_info);
hdr.coproc = TIM_COPROC;
hdr.msg = TIM_GET_DEV_INFO;
hdr.vfid = 0; /* TIM DEV is always 0. TIM RING ID changes. */
memset(info, 0, len);
return octeontx_mbox_send(&hdr, NULL, 0, info, len);
}
static void
timvf_ring_info_get(const struct rte_event_timer_adapter *adptr,
struct rte_event_timer_adapter_info *adptr_info)
{
struct timvf_ring *timr = adptr->data->adapter_priv;
adptr_info->max_tmo_ns = timr->max_tout;
adptr_info->min_resolution_ns = timr->tck_nsec;
rte_memcpy(&adptr_info->conf, &adptr->data->conf,
sizeof(struct rte_event_timer_adapter_conf));
}
static int
timvf_ring_conf_set(struct timvf_ctrl_reg *rctl, uint8_t ring_id)
{
struct octeontx_mbox_hdr hdr = {0};
uint16_t len = sizeof(struct timvf_ctrl_reg);
int ret;
hdr.coproc = TIM_COPROC;
hdr.msg = TIM_SET_RING_INFO;
hdr.vfid = ring_id;
ret = octeontx_mbox_send(&hdr, rctl, len, NULL, 0);
if (ret < 0 || hdr.res_code != MBOX_RET_SUCCESS)
return -EACCES;
return 0;
}
static int
timvf_get_start_cyc(uint64_t *now, uint8_t ring_id)
{
struct octeontx_mbox_hdr hdr = {0};
hdr.coproc = TIM_COPROC;
hdr.msg = TIM_RING_START_CYC_GET;
hdr.vfid = ring_id;
*now = 0;
return octeontx_mbox_send(&hdr, NULL, 0, now, sizeof(uint64_t));
}
static int
optimize_bucket_parameters(struct timvf_ring *timr)
{
uint32_t hbkts;
uint32_t lbkts;
uint64_t tck_nsec;
hbkts = rte_align32pow2(timr->nb_bkts);
tck_nsec = RTE_ALIGN_MUL_CEIL(timr->max_tout / (hbkts - 1), 10);
if ((tck_nsec < 1000 || hbkts > TIM_MAX_BUCKETS))
hbkts = 0;
lbkts = rte_align32prevpow2(timr->nb_bkts);
tck_nsec = RTE_ALIGN_MUL_CEIL((timr->max_tout / (lbkts - 1)), 10);
if ((tck_nsec < 1000 || hbkts > TIM_MAX_BUCKETS))
lbkts = 0;
if (!hbkts && !lbkts)
return 0;
if (!hbkts) {
timr->nb_bkts = lbkts;
goto end;
} else if (!lbkts) {
timr->nb_bkts = hbkts;
goto end;
}
timr->nb_bkts = (hbkts - timr->nb_bkts) <
(timr->nb_bkts - lbkts) ? hbkts : lbkts;
end:
timr->get_target_bkt = bkt_and;
timr->tck_nsec = RTE_ALIGN_MUL_CEIL((timr->max_tout /
(timr->nb_bkts - 1)), 10);
return 1;
}
static int
timvf_ring_start(const struct rte_event_timer_adapter *adptr)
{
int ret;
uint8_t use_fpa = 0;
uint64_t interval;
uintptr_t pool;
struct timvf_ctrl_reg rctrl;
struct timvf_mbox_dev_info dinfo;
struct timvf_ring *timr = adptr->data->adapter_priv;
ret = timvf_mbox_dev_info_get(&dinfo);
if (ret < 0 || ret != sizeof(struct timvf_mbox_dev_info))
return -EINVAL;
/* Calculate the interval cycles according to clock source. */
switch (timr->clk_src) {
case TIM_CLK_SRC_SCLK:
interval = NSEC2CLK(timr->tck_nsec, dinfo.clk_freq);
break;
case TIM_CLK_SRC_GPIO:
/* GPIO doesn't work on tck_nsec. */
interval = 0;
break;
case TIM_CLK_SRC_GTI:
interval = NSEC2CLK(timr->tck_nsec, dinfo.clk_freq);
break;
case TIM_CLK_SRC_PTP:
interval = NSEC2CLK(timr->tck_nsec, dinfo.clk_freq);
break;
default:
timvf_log_err("Unsupported clock source configured %d",
timr->clk_src);
return -EINVAL;
}
if (!strcmp(rte_mbuf_best_mempool_ops(), "octeontx_fpavf"))
use_fpa = 1;
/*CTRL0 register.*/
rctrl.rctrl0 = interval;
/*CTRL1 register.*/
rctrl.rctrl1 = (uint64_t)(timr->clk_src) << 51 |
1ull << 48 /* LOCK_EN (Enable hw bucket lock mechanism) */ |
1ull << 47 /* ENA */ |
1ull << 44 /* ENA_LDWB */ |
(timr->nb_bkts - 1);
rctrl.rctrl2 = (uint64_t)(TIM_CHUNK_SIZE / 16) << 40;
if (use_fpa) {
pool = (uintptr_t)((struct rte_mempool *)
timr->chunk_pool)->pool_id;
ret = octeontx_fpa_bufpool_gaura(pool);
if (ret < 0) {
timvf_log_dbg("Unable to get gaura id");
ret = -ENOMEM;
goto error;
}
timvf_write64((uint64_t)ret,
(uint8_t *)timr->vbar0 + TIM_VRING_AURA);
} else {
rctrl.rctrl1 |= 1ull << 43 /* ENA_DFB (Enable don't free) */;
}
timvf_write64((uintptr_t)timr->bkt,
(uint8_t *)timr->vbar0 + TIM_VRING_BASE);
timvf_set_chunk_refill(timr, use_fpa);
if (timvf_ring_conf_set(&rctrl, timr->tim_ring_id)) {
ret = -EACCES;
goto error;
}
if (timvf_get_start_cyc(&timr->ring_start_cyc,
timr->tim_ring_id) < 0) {
ret = -EACCES;
goto error;
}
timr->tck_int = NSEC2CLK(timr->tck_nsec, rte_get_timer_hz());
timr->fast_div = rte_reciprocal_value_u64(timr->tck_int);
timvf_log_info("nb_bkts %d min_ns %"PRIu64" min_cyc %"PRIu64""
" maxtmo %"PRIu64"\n",
timr->nb_bkts, timr->tck_nsec, interval,
timr->max_tout);
return 0;
error:
rte_free(timr->bkt);
rte_mempool_free(timr->chunk_pool);
return ret;
}
static int
timvf_ring_stop(const struct rte_event_timer_adapter *adptr)
{
struct timvf_ring *timr = adptr->data->adapter_priv;
struct timvf_ctrl_reg rctrl = {0};
rctrl.rctrl0 = timvf_read64((uint8_t *)timr->vbar0 + TIM_VRING_CTL0);
rctrl.rctrl1 = timvf_read64((uint8_t *)timr->vbar0 + TIM_VRING_CTL1);
rctrl.rctrl1 &= ~(1ull << 47); /* Disable */
rctrl.rctrl2 = timvf_read64((uint8_t *)timr->vbar0 + TIM_VRING_CTL2);
if (timvf_ring_conf_set(&rctrl, timr->tim_ring_id))
return -EACCES;
return 0;
}
static int
timvf_ring_create(struct rte_event_timer_adapter *adptr)
{
struct rte_event_timer_adapter_conf *rcfg = &adptr->data->conf;
uint16_t free_idx = UINT16_MAX;
unsigned int mp_flags = 0;
struct ssovf_evdev *edev;
struct timvf_ring *timr;
const char *mempool_ops;
uint8_t tim_ring_id;
char pool_name[25];
int i, ret;
tim_ring_id = timvf_get_ring();
if (tim_ring_id == UINT8_MAX)
return -ENODEV;
edev = ssovf_pmd_priv(event_dev);
timr = rte_zmalloc("octeontx_timvf_priv",
sizeof(struct timvf_ring), 0);
if (timr == NULL)
return -ENOMEM;
adptr->data->adapter_priv = timr;
/* Check config parameters. */
if ((rcfg->clk_src != RTE_EVENT_TIMER_ADAPTER_CPU_CLK) &&
(!rcfg->timer_tick_ns ||
rcfg->timer_tick_ns < TIM_MIN_INTERVAL)) {
timvf_log_err("Too low timer ticks");
goto cfg_err;
}
timr->clk_src = (int) rcfg->clk_src;
timr->tim_ring_id = tim_ring_id;
timr->tck_nsec = RTE_ALIGN_MUL_CEIL(rcfg->timer_tick_ns, 10);
timr->max_tout = rcfg->max_tmo_ns;
timr->nb_bkts = (timr->max_tout / timr->tck_nsec);
timr->vbar0 = timvf_bar(timr->tim_ring_id, 0);
timr->bkt_pos = (uint8_t *)timr->vbar0 + TIM_VRING_REL;
timr->nb_timers = rcfg->nb_timers;
timr->get_target_bkt = bkt_mod;
if (edev->available_events < timr->nb_timers) {
timvf_log_err(
"Max available events %"PRIu32" requested timer events %"PRIu64"",
edev->available_events, timr->nb_timers);
return -ENOMEM;
}
for (i = 0; i < edev->tim_ring_cnt; i++) {
if (edev->tim_ring_ids[i] == UINT16_MAX)
free_idx = i;
}
if (free_idx == UINT16_MAX) {
void *old_ptr;
edev->tim_ring_cnt++;
old_ptr = edev->tim_ring_ids;
edev->tim_ring_ids =
rte_realloc(edev->tim_ring_ids,
sizeof(uint16_t) * edev->tim_ring_cnt, 0);
if (edev->tim_ring_ids == NULL) {
edev->tim_ring_ids = old_ptr;
edev->tim_ring_cnt--;
return -ENOMEM;
}
edev->available_events -= timr->nb_timers;
} else {
edev->tim_ring_ids[free_idx] = tim_ring_id;
edev->available_events -= timr->nb_timers;
}
timr->nb_chunks = timr->nb_timers / nb_chunk_slots;
/* Try to optimize the bucket parameters. */
if ((rcfg->flags & RTE_EVENT_TIMER_ADAPTER_F_ADJUST_RES)
&& !rte_is_power_of_2(timr->nb_bkts)) {
if (optimize_bucket_parameters(timr)) {
timvf_log_info("Optimized configured values");
timvf_log_dbg("nb_bkts : %"PRIu32"", timr->nb_bkts);
timvf_log_dbg("tck_nsec : %"PRIu64"", timr->tck_nsec);
} else
timvf_log_info("Failed to Optimize configured values");
}
if (rcfg->flags & RTE_EVENT_TIMER_ADAPTER_F_SP_PUT) {
mp_flags = MEMPOOL_F_SP_PUT | MEMPOOL_F_SC_GET;
timvf_log_info("Using single producer mode");
}
timr->bkt = rte_zmalloc("octeontx_timvf_bucket",
(timr->nb_bkts) * sizeof(struct tim_mem_bucket),
0);
if (timr->bkt == NULL)
goto mem_err;
snprintf(pool_name, sizeof(pool_name), "timvf_chunk_pool%d",
timr->tim_ring_id);
timr->chunk_pool = (void *)rte_mempool_create_empty(pool_name,
timr->nb_chunks, TIM_CHUNK_SIZE, 0, 0, rte_socket_id(),
mp_flags);
if (!timr->chunk_pool) {
rte_free(timr->bkt);
timvf_log_err("Unable to create chunkpool.");
return -ENOMEM;
}
mempool_ops = rte_mbuf_best_mempool_ops();
ret = rte_mempool_set_ops_byname(timr->chunk_pool,
mempool_ops, NULL);
if (ret != 0) {
timvf_log_err("Unable to set chunkpool ops.");
goto mem_err;
}
ret = rte_mempool_populate_default(timr->chunk_pool);
if (ret < 0) {
timvf_log_err("Unable to set populate chunkpool.");
goto mem_err;
}
timvf_write64(0, (uint8_t *)timr->vbar0 + TIM_VRING_BASE);
timvf_write64(0, (uint8_t *)timr->vbar0 + TIM_VF_NRSPERR_INT);
timvf_write64(0, (uint8_t *)timr->vbar0 + TIM_VF_NRSPERR_INT_W1S);
timvf_write64(0x7, (uint8_t *)timr->vbar0 + TIM_VF_NRSPERR_ENA_W1C);
timvf_write64(0x7, (uint8_t *)timr->vbar0 + TIM_VF_NRSPERR_ENA_W1S);
return 0;
mem_err:
rte_free(timr);
return -ENOMEM;
cfg_err:
rte_free(timr);
return -EINVAL;
}
static int
timvf_ring_free(struct rte_event_timer_adapter *adptr)
{
struct timvf_ring *timr = adptr->data->adapter_priv;
struct ssovf_evdev *edev;
int i;
edev = ssovf_pmd_priv(event_dev);
for (i = 0; i < edev->tim_ring_cnt; i++) {
if (edev->tim_ring_ids[i] == timr->tim_ring_id) {
edev->available_events += timr->nb_timers;
edev->tim_ring_ids[i] = UINT16_MAX;
break;
}
}
rte_mempool_free(timr->chunk_pool);
rte_free(timr->bkt);
timvf_release_ring(timr->tim_ring_id);
rte_free(adptr->data->adapter_priv);
return 0;
}
static int
timvf_stats_get(const struct rte_event_timer_adapter *adapter,
struct rte_event_timer_adapter_stats *stats)
{
struct timvf_ring *timr = adapter->data->adapter_priv;
uint64_t bkt_cyc = rte_rdtsc() - timr->ring_start_cyc;
stats->evtim_exp_count = timr->tim_arm_cnt;
stats->ev_enq_count = timr->tim_arm_cnt;
stats->adapter_tick_count = rte_reciprocal_divide_u64(bkt_cyc,
&timr->fast_div);
return 0;
}
static int
timvf_stats_reset(const struct rte_event_timer_adapter *adapter)
{
struct timvf_ring *timr = adapter->data->adapter_priv;
timr->tim_arm_cnt = 0;
return 0;
}
static struct rte_event_timer_adapter_ops timvf_ops = {
.init = timvf_ring_create,
.uninit = timvf_ring_free,
.start = timvf_ring_start,
.stop = timvf_ring_stop,
.get_info = timvf_ring_info_get,
};
int
timvf_timer_adapter_caps_get(const struct rte_eventdev *dev, uint64_t flags,
uint32_t *caps, const struct rte_event_timer_adapter_ops **ops,
uint8_t enable_stats)
{
RTE_SET_USED(dev);
if (enable_stats) {
timvf_ops.stats_get = timvf_stats_get;
timvf_ops.stats_reset = timvf_stats_reset;
}
if (flags & RTE_EVENT_TIMER_ADAPTER_F_SP_PUT)
timvf_ops.arm_burst = enable_stats ?
timvf_timer_arm_burst_sp_stats :
timvf_timer_arm_burst_sp;
else
timvf_ops.arm_burst = enable_stats ?
timvf_timer_arm_burst_mp_stats :
timvf_timer_arm_burst_mp;
timvf_ops.arm_tmo_tick_burst = enable_stats ?
timvf_timer_arm_tmo_brst_stats :
timvf_timer_arm_tmo_brst;
timvf_ops.cancel_burst = timvf_timer_cancel_burst;
*caps = RTE_EVENT_TIMER_ADAPTER_CAP_INTERNAL_PORT;
*ops = &timvf_ops;
return 0;
}
void
timvf_set_eventdevice(struct rte_eventdev *dev)
{
event_dev = dev;
}