mirror of
https://github.com/AsahiLinux/u-boot
synced 2025-01-23 02:15:12 +00:00
e5ad79099c
Import cvmx-pko3-queue.c from 2013 U-Boot. It will be used by the later added drivers to support networking on the MIPS Octeon II / III platforms. Signed-off-by: Aaron Williams <awilliams@marvell.com> Signed-off-by: Stefan Roese <sr@denx.de>
879 lines
28 KiB
C
879 lines
28 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* Copyright (C) 2018-2022 Marvell International Ltd.
|
|
*/
|
|
|
|
#include <errno.h>
|
|
#include <log.h>
|
|
#include <time.h>
|
|
#include <linux/delay.h>
|
|
|
|
#include <mach/cvmx-regs.h>
|
|
#include <mach/cvmx-csr.h>
|
|
#include <mach/cvmx-bootmem.h>
|
|
#include <mach/octeon-model.h>
|
|
#include <mach/cvmx-fuse.h>
|
|
#include <mach/octeon-feature.h>
|
|
#include <mach/cvmx-qlm.h>
|
|
#include <mach/octeon_qlm.h>
|
|
#include <mach/cvmx-pcie.h>
|
|
#include <mach/cvmx-coremask.h>
|
|
|
|
#include <mach/cvmx-agl-defs.h>
|
|
#include <mach/cvmx-bgxx-defs.h>
|
|
#include <mach/cvmx-ciu-defs.h>
|
|
#include <mach/cvmx-gmxx-defs.h>
|
|
#include <mach/cvmx-gserx-defs.h>
|
|
#include <mach/cvmx-ilk-defs.h>
|
|
#include <mach/cvmx-ipd-defs.h>
|
|
#include <mach/cvmx-pcsx-defs.h>
|
|
#include <mach/cvmx-pcsxx-defs.h>
|
|
#include <mach/cvmx-pki-defs.h>
|
|
#include <mach/cvmx-pko-defs.h>
|
|
#include <mach/cvmx-xcv-defs.h>
|
|
|
|
#include <mach/cvmx-hwpko.h>
|
|
#include <mach/cvmx-ilk.h>
|
|
#include <mach/cvmx-pki.h>
|
|
#include <mach/cvmx-pko3.h>
|
|
#include <mach/cvmx-pko3-queue.h>
|
|
#include <mach/cvmx-pko3-resources.h>
|
|
|
|
#include <mach/cvmx-helper.h>
|
|
#include <mach/cvmx-helper-board.h>
|
|
#include <mach/cvmx-helper-cfg.h>
|
|
|
|
#include <mach/cvmx-helper-bgx.h>
|
|
#include <mach/cvmx-helper-cfg.h>
|
|
#include <mach/cvmx-helper-util.h>
|
|
#include <mach/cvmx-helper-pki.h>
|
|
|
|
/* Smalles Round-Robin quantum to use +1 */
|
|
#define CVMX_PKO3_RR_QUANTUM_MIN 0x10
|
|
|
|
static int debug; /* 1 for basic, 2 for detailed trace */
|
|
|
|
struct cvmx_pko3_dq {
|
|
unsigned dq_count : 6; /* Number of descriptor queues */
|
|
unsigned dq_base : 10; /* Descriptor queue start number */
|
|
#define CVMX_PKO3_SWIZZLE_IPD 0x0
|
|
};
|
|
|
|
/*
|
|
* @INTERNAL
|
|
* Descriptor Queue to IPD port mapping table.
|
|
*
|
|
* This pointer is per-core, contains the virtual address
|
|
* of a global named block which has 2^12 entries per each
|
|
* possible node.
|
|
*/
|
|
struct cvmx_pko3_dq *__cvmx_pko3_dq_table;
|
|
|
|
int cvmx_pko3_get_queue_base(int ipd_port)
|
|
{
|
|
struct cvmx_pko3_dq *dq_table;
|
|
int ret = -1;
|
|
unsigned int i;
|
|
struct cvmx_xport xp = cvmx_helper_ipd_port_to_xport(ipd_port);
|
|
|
|
/* get per-node table */
|
|
if (cvmx_unlikely(!__cvmx_pko3_dq_table))
|
|
__cvmx_pko3_dq_table_setup();
|
|
|
|
i = CVMX_PKO3_SWIZZLE_IPD ^ xp.port;
|
|
|
|
/* get per-node table */
|
|
dq_table = __cvmx_pko3_dq_table + CVMX_PKO3_IPD_NUM_MAX * xp.node;
|
|
|
|
if (cvmx_likely(dq_table[i].dq_count > 0))
|
|
ret = xp.node << 10 | dq_table[i].dq_base;
|
|
else if (debug)
|
|
cvmx_printf("ERROR: %s: no queues for ipd_port=%#x\n", __func__,
|
|
ipd_port);
|
|
|
|
return ret;
|
|
}
|
|
|
|
int cvmx_pko3_get_queue_num(int ipd_port)
|
|
{
|
|
struct cvmx_pko3_dq *dq_table;
|
|
int ret = -1;
|
|
unsigned int i;
|
|
struct cvmx_xport xp = cvmx_helper_ipd_port_to_xport(ipd_port);
|
|
|
|
/* get per-node table */
|
|
if (cvmx_unlikely(!__cvmx_pko3_dq_table))
|
|
__cvmx_pko3_dq_table_setup();
|
|
|
|
i = CVMX_PKO3_SWIZZLE_IPD ^ xp.port;
|
|
|
|
/* get per-node table */
|
|
dq_table = __cvmx_pko3_dq_table + CVMX_PKO3_IPD_NUM_MAX * xp.node;
|
|
|
|
if (cvmx_likely(dq_table[i].dq_count > 0))
|
|
ret = dq_table[i].dq_count;
|
|
else if (debug)
|
|
debug("ERROR: %s: no queues for ipd_port=%#x\n", __func__,
|
|
ipd_port);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @INTERNAL
|
|
*
|
|
* Initialize port/dq table contents
|
|
*/
|
|
static void __cvmx_pko3_dq_table_init(void *ptr)
|
|
{
|
|
unsigned int size = sizeof(struct cvmx_pko3_dq) *
|
|
CVMX_PKO3_IPD_NUM_MAX * CVMX_MAX_NODES;
|
|
|
|
memset(ptr, 0, size);
|
|
}
|
|
|
|
/**
|
|
* @INTERNAL
|
|
*
|
|
* Find or allocate global port/dq map table
|
|
* which is a named table, contains entries for
|
|
* all possible OCI nodes.
|
|
*
|
|
* The table global pointer is stored in core-local variable
|
|
* so that every core will call this function once, on first use.
|
|
*/
|
|
int __cvmx_pko3_dq_table_setup(void)
|
|
{
|
|
void *ptr;
|
|
|
|
ptr = cvmx_bootmem_alloc_named_range_once(
|
|
/* size */
|
|
sizeof(struct cvmx_pko3_dq) * CVMX_PKO3_IPD_NUM_MAX *
|
|
CVMX_MAX_NODES,
|
|
/* min_addr, max_addr, align */
|
|
0ull, 0ull, sizeof(struct cvmx_pko3_dq),
|
|
/* name */
|
|
"cvmx_pko3_global_dq_table", __cvmx_pko3_dq_table_init);
|
|
|
|
if (debug)
|
|
debug("%s: dq_table_ptr=%p\n", __func__, ptr);
|
|
|
|
if (!ptr)
|
|
return -1;
|
|
|
|
__cvmx_pko3_dq_table = ptr;
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* @INTERNAL
|
|
* Register a range of Descriptor Queues with an interface port
|
|
*
|
|
* This function populates the DQ-to-IPD translation table
|
|
* used by the application to retrieve the DQ range (typically ordered
|
|
* by priority) for a given IPD-port, which is either a physical port,
|
|
* or a channel on a channelized interface (i.e. ILK).
|
|
*
|
|
* @param xiface is the physical interface number
|
|
* @param index is either a physical port on an interface
|
|
* or a channel of an ILK interface
|
|
* @param dq_base is the first Descriptor Queue number in a consecutive range
|
|
* @param dq_count is the number of consecutive Descriptor Queues leading
|
|
* the same channel or port.
|
|
*
|
|
* Only a consecutive range of Descriptor Queues can be associated with any
|
|
* given channel/port, and usually they are ordered from most to least
|
|
* in terms of scheduling priority.
|
|
*
|
|
* Note: thus function only populates the node-local translation table.
|
|
* NOTE: This function would be cleaner if it had a single ipd_port argument
|
|
*
|
|
* @returns 0 on success, -1 on failure.
|
|
*/
|
|
int __cvmx_pko3_ipd_dq_register(int xiface, int index, unsigned int dq_base,
|
|
unsigned int dq_count)
|
|
{
|
|
struct cvmx_pko3_dq *dq_table;
|
|
int ipd_port;
|
|
unsigned int i;
|
|
struct cvmx_xiface xi = cvmx_helper_xiface_to_node_interface(xiface);
|
|
struct cvmx_xport xp;
|
|
|
|
if (__cvmx_helper_xiface_is_null(xiface)) {
|
|
ipd_port = cvmx_helper_node_to_ipd_port(xi.node,
|
|
CVMX_PKO3_IPD_PORT_NULL);
|
|
} else {
|
|
int p;
|
|
|
|
p = cvmx_helper_get_ipd_port(xiface, index);
|
|
if (p < 0) {
|
|
cvmx_printf("ERROR: %s: xiface %#x has no IPD port\n",
|
|
__func__, xiface);
|
|
return -1;
|
|
}
|
|
ipd_port = p;
|
|
}
|
|
|
|
xp = cvmx_helper_ipd_port_to_xport(ipd_port);
|
|
|
|
i = CVMX_PKO3_SWIZZLE_IPD ^ xp.port;
|
|
|
|
/* get per-node table */
|
|
if (!__cvmx_pko3_dq_table)
|
|
__cvmx_pko3_dq_table_setup();
|
|
|
|
dq_table = __cvmx_pko3_dq_table + CVMX_PKO3_IPD_NUM_MAX * xi.node;
|
|
|
|
if (debug)
|
|
debug("%s: ipd_port=%#x ix=%#x dq %u cnt %u\n", __func__,
|
|
ipd_port, i, dq_base, dq_count);
|
|
|
|
/* Check the IPD port has not already been configured */
|
|
if (dq_table[i].dq_count > 0) {
|
|
cvmx_printf("%s: ERROR: IPD %#x already registered\n", __func__,
|
|
ipd_port);
|
|
return -1;
|
|
}
|
|
|
|
/* Store DQ# range in the queue lookup table */
|
|
dq_table[i].dq_base = dq_base;
|
|
dq_table[i].dq_count = dq_count;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* @INTERNAL
|
|
* Convert normal CHAN_E (i.e. IPD port) value to compressed channel form
|
|
* that is used to populate PKO_LUT.
|
|
*
|
|
* Note: This code may be model specific.
|
|
*/
|
|
static int cvmx_pko3_chan_2_xchan(uint16_t ipd_port)
|
|
{
|
|
u16 xchan;
|
|
u8 off;
|
|
static const u8 *xchan_base;
|
|
static const u8 xchan_base_cn78xx[16] = {
|
|
/* IPD 0x000 */ 0x3c0 >> 4, /* LBK */
|
|
/* IPD 0x100 */ 0x380 >> 4, /* DPI */
|
|
/* IPD 0x200 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0x300 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0x400 */ 0x000 >> 4, /* ILK0 */
|
|
/* IPD 0x500 */ 0x100 >> 4, /* ILK1 */
|
|
/* IPD 0x600 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0x700 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0x800 */ 0x200 >> 4, /* BGX0 */
|
|
/* IPD 0x900 */ 0x240 >> 4, /* BGX1 */
|
|
/* IPD 0xa00 */ 0x280 >> 4, /* BGX2 */
|
|
/* IPD 0xb00 */ 0x2c0 >> 4, /* BGX3 */
|
|
/* IPD 0xc00 */ 0x300 >> 4, /* BGX4 */
|
|
/* IPD 0xd00 */ 0x340 >> 4, /* BGX5 */
|
|
/* IPD 0xe00 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0xf00 */ 0xfff >> 4 /* not used */
|
|
};
|
|
static const u8 xchan_base_cn73xx[16] = {
|
|
/* IPD 0x000 */ 0x0c0 >> 4, /* LBK */
|
|
/* IPD 0x100 */ 0x100 >> 4, /* DPI */
|
|
/* IPD 0x200 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0x300 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0x400 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0x500 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0x600 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0x700 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0x800 */ 0x000 >> 4, /* BGX0 */
|
|
/* IPD 0x900 */ 0x040 >> 4, /* BGX1 */
|
|
/* IPD 0xa00 */ 0x080 >> 4, /* BGX2 */
|
|
/* IPD 0xb00 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0xc00 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0xd00 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0xe00 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0xf00 */ 0xfff >> 4 /* not used */
|
|
};
|
|
static const u8 xchan_base_cn75xx[16] = {
|
|
/* IPD 0x000 */ 0x040 >> 4, /* LBK */
|
|
/* IPD 0x100 */ 0x080 >> 4, /* DPI */
|
|
/* IPD 0x200 */ 0xeee >> 4, /* SRIO0 noop */
|
|
/* IPD 0x300 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0x400 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0x500 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0x600 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0x700 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0x800 */ 0x000 >> 4, /* BGX0 */
|
|
/* IPD 0x900 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0xa00 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0xb00 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0xc00 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0xd00 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0xe00 */ 0xfff >> 4, /* not used */
|
|
/* IPD 0xf00 */ 0xfff >> 4 /* not used */
|
|
};
|
|
|
|
if (OCTEON_IS_MODEL(OCTEON_CN73XX))
|
|
xchan_base = xchan_base_cn73xx;
|
|
if (OCTEON_IS_MODEL(OCTEON_CNF75XX))
|
|
xchan_base = xchan_base_cn75xx;
|
|
if (OCTEON_IS_MODEL(OCTEON_CN78XX))
|
|
xchan_base = xchan_base_cn78xx;
|
|
|
|
if (!xchan_base)
|
|
return -1;
|
|
|
|
xchan = ipd_port >> 8;
|
|
|
|
/* ILKx, DPI has 8 bits logical channels, others just 6 */
|
|
if (((xchan & 0xfe) == 0x04) || xchan == 0x01)
|
|
off = ipd_port & 0xff;
|
|
else
|
|
off = ipd_port & 0x3f;
|
|
|
|
xchan = xchan_base[xchan & 0xf];
|
|
|
|
if (xchan == 0xff)
|
|
return -1; /* Invalid IPD_PORT */
|
|
else if (xchan == 0xee)
|
|
return -2; /* LUT not used */
|
|
else
|
|
return (xchan << 4) | off;
|
|
}
|
|
|
|
/*
|
|
* Map channel number in PKO
|
|
*
|
|
* @param node is to specify the node to which this configuration is applied.
|
|
* @param pq_num specifies the Port Queue (i.e. L1) queue number.
|
|
* @param l2_l3_q_num specifies L2/L3 queue number.
|
|
* @param channel specifies the channel number to map to the queue.
|
|
*
|
|
* The channel assignment applies to L2 or L3 Shaper Queues depending
|
|
* on the setting of channel credit level.
|
|
*
|
|
* @return returns none.
|
|
*/
|
|
void cvmx_pko3_map_channel(unsigned int node, unsigned int pq_num,
|
|
unsigned int l2_l3_q_num, uint16_t channel)
|
|
{
|
|
union cvmx_pko_l3_l2_sqx_channel sqx_channel;
|
|
cvmx_pko_lutx_t lutx;
|
|
int xchan;
|
|
|
|
sqx_channel.u64 =
|
|
csr_rd_node(node, CVMX_PKO_L3_L2_SQX_CHANNEL(l2_l3_q_num));
|
|
|
|
sqx_channel.s.cc_channel = channel;
|
|
|
|
csr_wr_node(node, CVMX_PKO_L3_L2_SQX_CHANNEL(l2_l3_q_num),
|
|
sqx_channel.u64);
|
|
|
|
/* Convert CHAN_E into compressed channel */
|
|
xchan = cvmx_pko3_chan_2_xchan(channel);
|
|
|
|
if (debug)
|
|
debug("%s: ipd_port=%#x xchan=%#x\n", __func__, channel, xchan);
|
|
|
|
if (xchan < 0) {
|
|
if (xchan == -1)
|
|
cvmx_printf("%s: ERROR: channel %#x not recognized\n",
|
|
__func__, channel);
|
|
return;
|
|
}
|
|
|
|
lutx.u64 = 0;
|
|
lutx.s.valid = 1;
|
|
lutx.s.pq_idx = pq_num;
|
|
lutx.s.queue_number = l2_l3_q_num;
|
|
|
|
csr_wr_node(node, CVMX_PKO_LUTX(xchan), lutx.u64);
|
|
|
|
if (debug)
|
|
debug("%s: channel %#x (compressed=%#x) mapped L2/L3 SQ=%u, PQ=%u\n",
|
|
__func__, channel, xchan, l2_l3_q_num, pq_num);
|
|
}
|
|
|
|
/*
|
|
* @INTERNAL
|
|
* This function configures port queue scheduling and topology parameters
|
|
* in hardware.
|
|
*
|
|
* @param node is to specify the node to which this configuration is applied.
|
|
* @param port_queue is the port queue number to be configured.
|
|
* @param mac_num is the mac number of the mac that will be tied to this port_queue.
|
|
*/
|
|
static void cvmx_pko_configure_port_queue(int node, int port_queue, int mac_num)
|
|
{
|
|
cvmx_pko_l1_sqx_topology_t pko_l1_topology;
|
|
cvmx_pko_l1_sqx_shape_t pko_l1_shape;
|
|
cvmx_pko_l1_sqx_link_t pko_l1_link;
|
|
|
|
pko_l1_topology.u64 = 0;
|
|
pko_l1_topology.s.link = mac_num;
|
|
csr_wr_node(node, CVMX_PKO_L1_SQX_TOPOLOGY(port_queue),
|
|
pko_l1_topology.u64);
|
|
|
|
pko_l1_shape.u64 = 0;
|
|
pko_l1_shape.s.link = mac_num;
|
|
csr_wr_node(node, CVMX_PKO_L1_SQX_SHAPE(port_queue), pko_l1_shape.u64);
|
|
|
|
pko_l1_link.u64 = 0;
|
|
pko_l1_link.s.link = mac_num;
|
|
csr_wr_node(node, CVMX_PKO_L1_SQX_LINK(port_queue), pko_l1_link.u64);
|
|
}
|
|
|
|
/*
|
|
* @INTERNAL
|
|
* This function configures level 2 queues scheduling and topology parameters
|
|
* in hardware.
|
|
*
|
|
* @param node is to specify the node to which this configuration is applied.
|
|
* @param queue is the level3 queue number to be configured.
|
|
* @param parent_queue is the parent queue at next level for this l3 queue.
|
|
* @param prio is this queue's priority in parent's scheduler.
|
|
* @param rr_quantum is this queue's round robin quantum value.
|
|
* @param child_base is the first child queue number in the static prioriy children.
|
|
* @param child_rr_prio is the round robin children priority.
|
|
*/
|
|
static void cvmx_pko_configure_l2_queue(int node, int queue, int parent_queue,
|
|
int prio, int rr_quantum,
|
|
int child_base, int child_rr_prio)
|
|
{
|
|
cvmx_pko_l2_sqx_schedule_t pko_sq_sched;
|
|
cvmx_pko_l2_sqx_topology_t pko_child_topology;
|
|
cvmx_pko_l1_sqx_topology_t pko_parent_topology;
|
|
|
|
/* parent topology configuration */
|
|
pko_parent_topology.u64 =
|
|
csr_rd_node(node, CVMX_PKO_L1_SQX_TOPOLOGY(parent_queue));
|
|
pko_parent_topology.s.prio_anchor = child_base;
|
|
pko_parent_topology.s.rr_prio = child_rr_prio;
|
|
csr_wr_node(node, CVMX_PKO_L1_SQX_TOPOLOGY(parent_queue),
|
|
pko_parent_topology.u64);
|
|
|
|
if (debug > 1)
|
|
debug("CVMX_PKO_L1_SQX_TOPOLOGY(%u): PRIO_ANCHOR=%u PARENT=%u\n",
|
|
parent_queue, pko_parent_topology.s.prio_anchor,
|
|
pko_parent_topology.s.link);
|
|
|
|
/* scheduler configuration for this sq in the parent queue */
|
|
pko_sq_sched.u64 = 0;
|
|
pko_sq_sched.s.prio = prio;
|
|
pko_sq_sched.s.rr_quantum = rr_quantum;
|
|
csr_wr_node(node, CVMX_PKO_L2_SQX_SCHEDULE(queue), pko_sq_sched.u64);
|
|
|
|
/* child topology configuration */
|
|
pko_child_topology.u64 = 0;
|
|
pko_child_topology.s.parent = parent_queue;
|
|
csr_wr_node(node, CVMX_PKO_L2_SQX_TOPOLOGY(queue),
|
|
pko_child_topology.u64);
|
|
}
|
|
|
|
/*
|
|
* @INTERNAL
|
|
* This function configures level 3 queues scheduling and topology parameters
|
|
* in hardware.
|
|
*
|
|
* @param node is to specify the node to which this configuration is applied.
|
|
* @param queue is the level3 queue number to be configured.
|
|
* @param parent_queue is the parent queue at next level for this l3 queue.
|
|
* @param prio is this queue's priority in parent's scheduler.
|
|
* @param rr_quantum is this queue's round robin quantum value.
|
|
* @param child_base is the first child queue number in the static prioriy children.
|
|
* @param child_rr_prio is the round robin children priority.
|
|
*/
|
|
static void cvmx_pko_configure_l3_queue(int node, int queue, int parent_queue,
|
|
int prio, int rr_quantum,
|
|
int child_base, int child_rr_prio)
|
|
{
|
|
cvmx_pko_l3_sqx_schedule_t pko_sq_sched;
|
|
cvmx_pko_l3_sqx_topology_t pko_child_topology;
|
|
cvmx_pko_l2_sqx_topology_t pko_parent_topology;
|
|
|
|
/* parent topology configuration */
|
|
pko_parent_topology.u64 =
|
|
csr_rd_node(node, CVMX_PKO_L2_SQX_TOPOLOGY(parent_queue));
|
|
pko_parent_topology.s.prio_anchor = child_base;
|
|
pko_parent_topology.s.rr_prio = child_rr_prio;
|
|
csr_wr_node(node, CVMX_PKO_L2_SQX_TOPOLOGY(parent_queue),
|
|
pko_parent_topology.u64);
|
|
|
|
if (debug > 1)
|
|
debug("CVMX_PKO_L2_SQX_TOPOLOGY(%u): PRIO_ANCHOR=%u PARENT=%u\n",
|
|
parent_queue, pko_parent_topology.s.prio_anchor,
|
|
pko_parent_topology.s.parent);
|
|
|
|
/* scheduler configuration for this sq in the parent queue */
|
|
pko_sq_sched.u64 = 0;
|
|
pko_sq_sched.s.prio = prio;
|
|
pko_sq_sched.s.rr_quantum = rr_quantum;
|
|
csr_wr_node(node, CVMX_PKO_L3_SQX_SCHEDULE(queue), pko_sq_sched.u64);
|
|
|
|
/* child topology configuration */
|
|
pko_child_topology.u64 = 0;
|
|
pko_child_topology.s.parent = parent_queue;
|
|
csr_wr_node(node, CVMX_PKO_L3_SQX_TOPOLOGY(queue),
|
|
pko_child_topology.u64);
|
|
}
|
|
|
|
/*
|
|
* @INTERNAL
|
|
* This function configures level 4 queues scheduling and topology parameters
|
|
* in hardware.
|
|
*
|
|
* @param node is to specify the node to which this configuration is applied.
|
|
* @param queue is the level4 queue number to be configured.
|
|
* @param parent_queue is the parent queue at next level for this l4 queue.
|
|
* @param prio is this queue's priority in parent's scheduler.
|
|
* @param rr_quantum is this queue's round robin quantum value.
|
|
* @param child_base is the first child queue number in the static prioriy children.
|
|
* @param child_rr_prio is the round robin children priority.
|
|
*/
|
|
static void cvmx_pko_configure_l4_queue(int node, int queue, int parent_queue,
|
|
int prio, int rr_quantum,
|
|
int child_base, int child_rr_prio)
|
|
{
|
|
cvmx_pko_l4_sqx_schedule_t pko_sq_sched;
|
|
cvmx_pko_l4_sqx_topology_t pko_child_topology;
|
|
cvmx_pko_l3_sqx_topology_t pko_parent_topology;
|
|
|
|
/* parent topology configuration */
|
|
pko_parent_topology.u64 =
|
|
csr_rd_node(node, CVMX_PKO_L3_SQX_TOPOLOGY(parent_queue));
|
|
pko_parent_topology.s.prio_anchor = child_base;
|
|
pko_parent_topology.s.rr_prio = child_rr_prio;
|
|
csr_wr_node(node, CVMX_PKO_L3_SQX_TOPOLOGY(parent_queue),
|
|
pko_parent_topology.u64);
|
|
|
|
if (debug > 1)
|
|
debug("CVMX_PKO_L3_SQX_TOPOLOGY(%u): PRIO_ANCHOR=%u PARENT=%u\n",
|
|
parent_queue, pko_parent_topology.s.prio_anchor,
|
|
pko_parent_topology.s.parent);
|
|
|
|
/* scheduler configuration for this sq in the parent queue */
|
|
pko_sq_sched.u64 = 0;
|
|
pko_sq_sched.s.prio = prio;
|
|
pko_sq_sched.s.rr_quantum = rr_quantum;
|
|
csr_wr_node(node, CVMX_PKO_L4_SQX_SCHEDULE(queue), pko_sq_sched.u64);
|
|
|
|
/* topology configuration */
|
|
pko_child_topology.u64 = 0;
|
|
pko_child_topology.s.parent = parent_queue;
|
|
csr_wr_node(node, CVMX_PKO_L4_SQX_TOPOLOGY(queue),
|
|
pko_child_topology.u64);
|
|
}
|
|
|
|
/*
|
|
* @INTERNAL
|
|
* This function configures level 5 queues scheduling and topology parameters
|
|
* in hardware.
|
|
*
|
|
* @param node is to specify the node to which this configuration is applied.
|
|
* @param queue is the level5 queue number to be configured.
|
|
* @param parent_queue is the parent queue at next level for this l5 queue.
|
|
* @param prio is this queue's priority in parent's scheduler.
|
|
* @param rr_quantum is this queue's round robin quantum value.
|
|
* @param child_base is the first child queue number in the static prioriy children.
|
|
* @param child_rr_prio is the round robin children priority.
|
|
*/
|
|
static void cvmx_pko_configure_l5_queue(int node, int queue, int parent_queue,
|
|
int prio, int rr_quantum,
|
|
int child_base, int child_rr_prio)
|
|
{
|
|
cvmx_pko_l5_sqx_schedule_t pko_sq_sched;
|
|
cvmx_pko_l4_sqx_topology_t pko_parent_topology;
|
|
cvmx_pko_l5_sqx_topology_t pko_child_topology;
|
|
|
|
/* parent topology configuration */
|
|
pko_parent_topology.u64 =
|
|
csr_rd_node(node, CVMX_PKO_L4_SQX_TOPOLOGY(parent_queue));
|
|
pko_parent_topology.s.prio_anchor = child_base;
|
|
pko_parent_topology.s.rr_prio = child_rr_prio;
|
|
csr_wr_node(node, CVMX_PKO_L4_SQX_TOPOLOGY(parent_queue),
|
|
pko_parent_topology.u64);
|
|
|
|
if (debug > 1)
|
|
debug("CVMX_PKO_L4_SQX_TOPOLOGY(%u): PRIO_ANCHOR=%u PARENT=%u\n",
|
|
parent_queue, pko_parent_topology.s.prio_anchor,
|
|
pko_parent_topology.s.parent);
|
|
|
|
/* scheduler configuration for this sq in the parent queue */
|
|
pko_sq_sched.u64 = 0;
|
|
pko_sq_sched.s.prio = prio;
|
|
pko_sq_sched.s.rr_quantum = rr_quantum;
|
|
csr_wr_node(node, CVMX_PKO_L5_SQX_SCHEDULE(queue), pko_sq_sched.u64);
|
|
|
|
/* topology configuration */
|
|
pko_child_topology.u64 = 0;
|
|
pko_child_topology.s.parent = parent_queue;
|
|
csr_wr_node(node, CVMX_PKO_L5_SQX_TOPOLOGY(queue),
|
|
pko_child_topology.u64);
|
|
}
|
|
|
|
/*
|
|
* @INTERNAL
|
|
* This function configures descriptor queues scheduling and topology parameters
|
|
* in hardware.
|
|
*
|
|
* @param node is to specify the node to which this configuration is applied.
|
|
* @param dq is the descriptor queue number to be configured.
|
|
* @param parent_queue is the parent queue at next level for this dq.
|
|
* @param prio is this queue's priority in parent's scheduler.
|
|
* @param rr_quantum is this queue's round robin quantum value.
|
|
* @param child_base is the first child queue number in the static prioriy children.
|
|
* @param child_rr_prio is the round robin children priority.
|
|
*/
|
|
static void cvmx_pko_configure_dq(int node, int dq, int parent_queue, int prio,
|
|
int rr_quantum, int child_base,
|
|
int child_rr_prio)
|
|
{
|
|
cvmx_pko_dqx_schedule_t pko_dq_sched;
|
|
cvmx_pko_dqx_topology_t pko_dq_topology;
|
|
cvmx_pko_l5_sqx_topology_t pko_parent_topology;
|
|
cvmx_pko_dqx_wm_ctl_t pko_dq_wm_ctl;
|
|
unsigned long long parent_topology_reg;
|
|
char lvl;
|
|
|
|
if (debug)
|
|
debug("%s: dq %u parent %u child_base %u\n", __func__, dq,
|
|
parent_queue, child_base);
|
|
|
|
if (__cvmx_pko3_sq_lvl_max() == CVMX_PKO_L5_QUEUES) {
|
|
parent_topology_reg = CVMX_PKO_L5_SQX_TOPOLOGY(parent_queue);
|
|
lvl = 5;
|
|
} else if (__cvmx_pko3_sq_lvl_max() == CVMX_PKO_L3_QUEUES) {
|
|
parent_topology_reg = CVMX_PKO_L3_SQX_TOPOLOGY(parent_queue);
|
|
lvl = 3;
|
|
} else {
|
|
return;
|
|
}
|
|
|
|
if (debug)
|
|
debug("%s: parent_topology_reg=%#llx\n", __func__,
|
|
parent_topology_reg);
|
|
|
|
/* parent topology configuration */
|
|
pko_parent_topology.u64 = csr_rd_node(node, parent_topology_reg);
|
|
pko_parent_topology.s.prio_anchor = child_base;
|
|
pko_parent_topology.s.rr_prio = child_rr_prio;
|
|
csr_wr_node(node, parent_topology_reg, pko_parent_topology.u64);
|
|
|
|
if (debug > 1)
|
|
debug("CVMX_PKO_L%d_SQX_TOPOLOGY(%u): PRIO_ANCHOR=%u PARENT=%u\n",
|
|
lvl, parent_queue, pko_parent_topology.s.prio_anchor,
|
|
pko_parent_topology.s.parent);
|
|
|
|
/* scheduler configuration for this dq in the parent queue */
|
|
pko_dq_sched.u64 = 0;
|
|
pko_dq_sched.s.prio = prio;
|
|
pko_dq_sched.s.rr_quantum = rr_quantum;
|
|
csr_wr_node(node, CVMX_PKO_DQX_SCHEDULE(dq), pko_dq_sched.u64);
|
|
|
|
/* topology configuration */
|
|
pko_dq_topology.u64 = 0;
|
|
pko_dq_topology.s.parent = parent_queue;
|
|
csr_wr_node(node, CVMX_PKO_DQX_TOPOLOGY(dq), pko_dq_topology.u64);
|
|
|
|
/* configure for counting packets, not bytes at this level */
|
|
pko_dq_wm_ctl.u64 = 0;
|
|
pko_dq_wm_ctl.s.kind = 1;
|
|
pko_dq_wm_ctl.s.enable = 0;
|
|
csr_wr_node(node, CVMX_PKO_DQX_WM_CTL(dq), pko_dq_wm_ctl.u64);
|
|
|
|
if (debug > 1) {
|
|
pko_dq_sched.u64 = csr_rd_node(node, CVMX_PKO_DQX_SCHEDULE(dq));
|
|
pko_dq_topology.u64 =
|
|
csr_rd_node(node, CVMX_PKO_DQX_TOPOLOGY(dq));
|
|
debug("CVMX_PKO_DQX_TOPOLOGY(%u)PARENT=%u CVMX_PKO_DQX_SCHEDULE(%u) PRIO=%u Q=%u\n",
|
|
dq, pko_dq_topology.s.parent, dq, pko_dq_sched.s.prio,
|
|
pko_dq_sched.s.rr_quantum);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* @INTERNAL
|
|
* The following structure selects the Scheduling Queue configuration
|
|
* routine for each of the supported levels.
|
|
* The initial content of the table will be setup in accordance
|
|
* to the specific SoC model and its implemented resources
|
|
*/
|
|
struct pko3_cfg_tab_s {
|
|
/* function pointer for to configure the given level, last=DQ */
|
|
struct {
|
|
u8 parent_level;
|
|
void (*cfg_sq_func)(int node, int queue, int parent_queue,
|
|
int prio, int rr_quantum, int child_base,
|
|
int child_rr_prio);
|
|
//XXX for debugging exagerated size
|
|
} lvl[256];
|
|
};
|
|
|
|
static const struct pko3_cfg_tab_s pko3_cn78xx_cfg = {
|
|
{ [CVMX_PKO_L2_QUEUES] = { CVMX_PKO_PORT_QUEUES,
|
|
cvmx_pko_configure_l2_queue },
|
|
[CVMX_PKO_L3_QUEUES] = { CVMX_PKO_L2_QUEUES,
|
|
cvmx_pko_configure_l3_queue },
|
|
[CVMX_PKO_L4_QUEUES] = { CVMX_PKO_L3_QUEUES,
|
|
cvmx_pko_configure_l4_queue },
|
|
[CVMX_PKO_L5_QUEUES] = { CVMX_PKO_L4_QUEUES,
|
|
cvmx_pko_configure_l5_queue },
|
|
[CVMX_PKO_DESCR_QUEUES] = { CVMX_PKO_L5_QUEUES,
|
|
cvmx_pko_configure_dq } }
|
|
};
|
|
|
|
static const struct pko3_cfg_tab_s pko3_cn73xx_cfg = {
|
|
{ [CVMX_PKO_L2_QUEUES] = { CVMX_PKO_PORT_QUEUES,
|
|
cvmx_pko_configure_l2_queue },
|
|
[CVMX_PKO_L3_QUEUES] = { CVMX_PKO_L2_QUEUES,
|
|
cvmx_pko_configure_l3_queue },
|
|
[CVMX_PKO_DESCR_QUEUES] = { CVMX_PKO_L3_QUEUES,
|
|
cvmx_pko_configure_dq } }
|
|
};
|
|
|
|
/*
|
|
* Configure Port Queue and its children Scheduler Queue
|
|
*
|
|
* Port Queues (a.k.a L1) are assigned 1-to-1 to MACs.
|
|
* L2 Scheduler Queues are used for specifying channels, and thus there
|
|
* could be multiple L2 SQs attached to a single L1 PQ, either in a
|
|
* fair round-robin scheduling, or with static and/or round-robin priorities.
|
|
*
|
|
* @param node on which to operate
|
|
* @param mac_num is the LMAC number to that is associated with the Port Queue,
|
|
* @param pq_num is the number of the L1 PQ attached to the MAC
|
|
*
|
|
* @returns 0 on success, -1 on failure.
|
|
*/
|
|
int cvmx_pko3_pq_config(unsigned int node, unsigned int mac_num,
|
|
unsigned int pq_num)
|
|
{
|
|
char b1[10];
|
|
|
|
if (debug)
|
|
debug("%s: MAC%u -> %s\n", __func__, mac_num,
|
|
__cvmx_pko3_sq_str(b1, CVMX_PKO_PORT_QUEUES, pq_num));
|
|
|
|
cvmx_pko_configure_port_queue(node, pq_num, mac_num);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Configure L3 through L5 Scheduler Queues and Descriptor Queues
|
|
*
|
|
* The Scheduler Queues in Levels 3 to 5 and Descriptor Queues are
|
|
* configured one-to-one or many-to-one to a single parent Scheduler
|
|
* Queues. The level of the parent SQ is specified in an argument,
|
|
* as well as the number of childer to attach to the specific parent.
|
|
* The children can have fair round-robin or priority-based scheduling
|
|
* when multiple children are assigned a single parent.
|
|
*
|
|
* @param node on which to operate
|
|
* @param child_level is the level of the child queue
|
|
* @param parent_queue is the number of the parent Scheduler Queue
|
|
* @param child_base is the number of the first child SQ or DQ to assign to
|
|
* @param child_count is the number of consecutive children to assign
|
|
* @param stat_prio_count is the priority setting for the children L2 SQs
|
|
*
|
|
* If <stat_prio_count> is -1, the Ln children will have equal Round-Robin
|
|
* relationship with eachother. If <stat_prio_count> is 0, all Ln children
|
|
* will be arranged in Weighted-Round-Robin, with the first having the most
|
|
* precedence. If <stat_prio_count> is between 1 and 8, it indicates how
|
|
* many children will have static priority settings (with the first having
|
|
* the most precedence), with the remaining Ln children having WRR scheduling.
|
|
*
|
|
* @returns 0 on success, -1 on failure.
|
|
*
|
|
* Note: this function supports the configuration of node-local unit.
|
|
*/
|
|
int cvmx_pko3_sq_config_children(unsigned int node,
|
|
enum cvmx_pko3_level_e child_level,
|
|
unsigned int parent_queue,
|
|
unsigned int child_base,
|
|
unsigned int child_count, int stat_prio_count)
|
|
{
|
|
enum cvmx_pko3_level_e parent_level;
|
|
unsigned int num_elem = 0;
|
|
unsigned int rr_quantum, rr_count;
|
|
unsigned int child, prio, rr_prio;
|
|
const struct pko3_cfg_tab_s *cfg_tbl = NULL;
|
|
char b1[10], b2[10];
|
|
|
|
if (OCTEON_IS_MODEL(OCTEON_CN78XX)) {
|
|
num_elem = NUM_ELEMENTS(pko3_cn78xx_cfg.lvl);
|
|
cfg_tbl = &pko3_cn78xx_cfg;
|
|
}
|
|
if (OCTEON_IS_MODEL(OCTEON_CN73XX) || OCTEON_IS_MODEL(OCTEON_CNF75XX)) {
|
|
num_elem = NUM_ELEMENTS(pko3_cn73xx_cfg.lvl);
|
|
cfg_tbl = &pko3_cn73xx_cfg;
|
|
}
|
|
|
|
if (!cfg_tbl || child_level >= num_elem) {
|
|
cvmx_printf("ERROR: %s: model or level %#x invalid\n", __func__,
|
|
child_level);
|
|
return -1;
|
|
}
|
|
|
|
parent_level = cfg_tbl->lvl[child_level].parent_level;
|
|
|
|
if (!cfg_tbl->lvl[child_level].cfg_sq_func ||
|
|
cfg_tbl->lvl[child_level].parent_level == 0) {
|
|
cvmx_printf("ERROR: %s: queue level %#x invalid\n", __func__,
|
|
child_level);
|
|
return -1;
|
|
}
|
|
|
|
/* First static priority is 0 - top precedence */
|
|
prio = 0;
|
|
|
|
if (stat_prio_count > (signed int)child_count)
|
|
stat_prio_count = child_count;
|
|
|
|
/* Valid PRIO field is 0..9, limit maximum static priorities */
|
|
if (stat_prio_count > 9)
|
|
stat_prio_count = 9;
|
|
|
|
/* Special case of a single child */
|
|
if (child_count == 1) {
|
|
rr_count = 0;
|
|
rr_prio = 0xF;
|
|
/* Special case for Fair-RR */
|
|
} else if (stat_prio_count < 0) {
|
|
rr_count = child_count;
|
|
rr_prio = 0;
|
|
} else {
|
|
rr_count = child_count - stat_prio_count;
|
|
rr_prio = stat_prio_count;
|
|
}
|
|
|
|
/* Compute highest RR_QUANTUM */
|
|
if (stat_prio_count > 0)
|
|
rr_quantum = CVMX_PKO3_RR_QUANTUM_MIN * rr_count;
|
|
else
|
|
rr_quantum = CVMX_PKO3_RR_QUANTUM_MIN;
|
|
|
|
if (debug)
|
|
debug("%s: Parent %s child_base %u rr_pri %u\n", __func__,
|
|
__cvmx_pko3_sq_str(b1, parent_level, parent_queue),
|
|
child_base, rr_prio);
|
|
|
|
/* Parent is configured with child */
|
|
|
|
for (child = child_base; child < (child_base + child_count); child++) {
|
|
if (debug)
|
|
debug("%s: Child %s of %s prio %u rr_quantum %#x\n",
|
|
__func__,
|
|
__cvmx_pko3_sq_str(b1, child_level, child),
|
|
__cvmx_pko3_sq_str(b2, parent_level,
|
|
parent_queue),
|
|
prio, rr_quantum);
|
|
|
|
cfg_tbl->lvl[child_level].cfg_sq_func(node, child, parent_queue,
|
|
prio, rr_quantum,
|
|
child_base, rr_prio);
|
|
|
|
if (prio < rr_prio)
|
|
prio++;
|
|
else if (stat_prio_count > 0)
|
|
rr_quantum -= CVMX_PKO3_RR_QUANTUM_MIN;
|
|
} /* for child */
|
|
|
|
return 0;
|
|
}
|