/* SPDX-License-Identifier: BSD-3-Clause * Copyright(c) 2017 Cavium, Inc */ #include "timvf_evdev.h" int otx_logtype_timvf; RTE_INIT(otx_timvf_init_log) { otx_logtype_timvf = rte_log_register("pmd.event.octeontx.timer"); if (otx_logtype_timvf >= 0) rte_log_set_level(otx_logtype_timvf, RTE_LOG_NOTICE); } 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) { char pool_name[25]; int ret; uint8_t tim_ring_id; uint64_t nb_timers; struct rte_event_timer_adapter_conf *rcfg = &adptr->data->conf; struct timvf_ring *timr; const char *mempool_ops; unsigned int mp_flags = 0; tim_ring_id = timvf_get_ring(); if (tim_ring_id == UINT8_MAX) return -ENODEV; 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; nb_timers = rcfg->nb_timers; timr->get_target_bkt = bkt_mod; timr->nb_chunks = 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; 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; }