| // SPDX-License-Identifier: GPL-2.0+ |
| /* Microchip Sparx5 Switch driver |
| * |
| * Copyright (c) 2023 Microchip Technology Inc. and its subsidiaries. |
| */ |
| |
| #include "sparx5_main_regs.h" |
| #include "sparx5_main.h" |
| |
| struct sparx5_sdlb_group sdlb_groups[SPX5_SDLB_GROUP_CNT] = { |
| { SPX5_SDLB_GROUP_RATE_MAX, 8192 / 1, 64 }, /* 25 G */ |
| { 15000000000ULL, 8192 / 1, 64 }, /* 15 G */ |
| { 10000000000ULL, 8192 / 1, 64 }, /* 10 G */ |
| { 5000000000ULL, 8192 / 1, 64 }, /* 5 G */ |
| { 2500000000ULL, 8192 / 1, 64 }, /* 2.5 G */ |
| { 1000000000ULL, 8192 / 2, 64 }, /* 1 G */ |
| { 500000000ULL, 8192 / 2, 64 }, /* 500 M */ |
| { 100000000ULL, 8192 / 4, 64 }, /* 100 M */ |
| { 50000000ULL, 8192 / 4, 64 }, /* 50 M */ |
| { 5000000ULL, 8192 / 8, 64 } /* 5 M */ |
| }; |
| |
| int sparx5_sdlb_clk_hz_get(struct sparx5 *sparx5) |
| { |
| u32 clk_per_100ps; |
| u64 clk_hz; |
| |
| clk_per_100ps = HSCH_SYS_CLK_PER_100PS_GET(spx5_rd(sparx5, |
| HSCH_SYS_CLK_PER)); |
| if (!clk_per_100ps) |
| clk_per_100ps = SPX5_CLK_PER_100PS_DEFAULT; |
| |
| clk_hz = (10 * 1000 * 1000) / clk_per_100ps; |
| return clk_hz *= 1000; |
| } |
| |
| static int sparx5_sdlb_pup_interval_get(struct sparx5 *sparx5, u32 max_token, |
| u64 max_rate) |
| { |
| u64 clk_hz; |
| |
| clk_hz = sparx5_sdlb_clk_hz_get(sparx5); |
| |
| return div64_u64((8 * clk_hz * max_token), max_rate); |
| } |
| |
| int sparx5_sdlb_pup_token_get(struct sparx5 *sparx5, u32 pup_interval, u64 rate) |
| { |
| u64 clk_hz; |
| |
| if (!rate) |
| return SPX5_SDLB_PUP_TOKEN_DISABLE; |
| |
| clk_hz = sparx5_sdlb_clk_hz_get(sparx5); |
| |
| return DIV64_U64_ROUND_UP((rate * pup_interval), (clk_hz * 8)); |
| } |
| |
| static void sparx5_sdlb_group_disable(struct sparx5 *sparx5, u32 group) |
| { |
| spx5_rmw(ANA_AC_SDLB_PUP_CTRL_PUP_ENA_SET(0), |
| ANA_AC_SDLB_PUP_CTRL_PUP_ENA, sparx5, |
| ANA_AC_SDLB_PUP_CTRL(group)); |
| } |
| |
| static void sparx5_sdlb_group_enable(struct sparx5 *sparx5, u32 group) |
| { |
| spx5_rmw(ANA_AC_SDLB_PUP_CTRL_PUP_ENA_SET(1), |
| ANA_AC_SDLB_PUP_CTRL_PUP_ENA, sparx5, |
| ANA_AC_SDLB_PUP_CTRL(group)); |
| } |
| |
| static u32 sparx5_sdlb_group_get_first(struct sparx5 *sparx5, u32 group) |
| { |
| u32 val; |
| |
| val = spx5_rd(sparx5, ANA_AC_SDLB_XLB_START(group)); |
| |
| return ANA_AC_SDLB_XLB_START_LBSET_START_GET(val); |
| } |
| |
| static u32 sparx5_sdlb_group_get_next(struct sparx5 *sparx5, u32 group, |
| u32 lb) |
| { |
| u32 val; |
| |
| val = spx5_rd(sparx5, ANA_AC_SDLB_XLB_NEXT(lb)); |
| |
| return ANA_AC_SDLB_XLB_NEXT_LBSET_NEXT_GET(val); |
| } |
| |
| static bool sparx5_sdlb_group_is_first(struct sparx5 *sparx5, u32 group, |
| u32 lb) |
| { |
| return lb == sparx5_sdlb_group_get_first(sparx5, group); |
| } |
| |
| static bool sparx5_sdlb_group_is_last(struct sparx5 *sparx5, u32 group, |
| u32 lb) |
| { |
| return lb == sparx5_sdlb_group_get_next(sparx5, group, lb); |
| } |
| |
| static bool sparx5_sdlb_group_is_empty(struct sparx5 *sparx5, u32 group) |
| { |
| u32 val; |
| |
| val = spx5_rd(sparx5, ANA_AC_SDLB_PUP_CTRL(group)); |
| |
| return ANA_AC_SDLB_PUP_CTRL_PUP_ENA_GET(val) == 0; |
| } |
| |
| static u32 sparx5_sdlb_group_get_last(struct sparx5 *sparx5, u32 group) |
| { |
| u32 itr, next; |
| |
| itr = sparx5_sdlb_group_get_first(sparx5, group); |
| |
| for (;;) { |
| next = sparx5_sdlb_group_get_next(sparx5, group, itr); |
| if (itr == next) |
| return itr; |
| |
| itr = next; |
| } |
| } |
| |
| static bool sparx5_sdlb_group_is_singular(struct sparx5 *sparx5, u32 group) |
| { |
| if (sparx5_sdlb_group_is_empty(sparx5, group)) |
| return false; |
| |
| return sparx5_sdlb_group_get_first(sparx5, group) == |
| sparx5_sdlb_group_get_last(sparx5, group); |
| } |
| |
| static int sparx5_sdlb_group_get_adjacent(struct sparx5 *sparx5, u32 group, |
| u32 idx, u32 *prev, u32 *next, |
| u32 *first) |
| { |
| u32 itr; |
| |
| *first = sparx5_sdlb_group_get_first(sparx5, group); |
| *prev = *first; |
| *next = *first; |
| itr = *first; |
| |
| for (;;) { |
| *next = sparx5_sdlb_group_get_next(sparx5, group, itr); |
| |
| if (itr == idx) |
| return 0; /* Found it */ |
| |
| if (itr == *next) |
| return -EINVAL; /* Was not found */ |
| |
| *prev = itr; |
| itr = *next; |
| } |
| } |
| |
| static int sparx5_sdlb_group_get_count(struct sparx5 *sparx5, u32 group) |
| { |
| u32 itr, next; |
| int count = 0; |
| |
| itr = sparx5_sdlb_group_get_first(sparx5, group); |
| |
| for (;;) { |
| next = sparx5_sdlb_group_get_next(sparx5, group, itr); |
| if (itr == next) |
| return count; |
| |
| itr = next; |
| count++; |
| } |
| } |
| |
| int sparx5_sdlb_group_get_by_rate(struct sparx5 *sparx5, u32 rate, u32 burst) |
| { |
| const struct sparx5_sdlb_group *group; |
| u64 rate_bps; |
| int i, count; |
| |
| rate_bps = rate * 1000; |
| |
| for (i = SPX5_SDLB_GROUP_CNT - 1; i >= 0; i--) { |
| group = &sdlb_groups[i]; |
| |
| count = sparx5_sdlb_group_get_count(sparx5, i); |
| |
| /* Check that this group is not full. |
| * According to LB group configuration rules: the number of XLBs |
| * in a group must not exceed PUP_INTERVAL/4 - 1. |
| */ |
| if (count > ((group->pup_interval / 4) - 1)) |
| continue; |
| |
| if (rate_bps < group->max_rate) |
| return i; |
| } |
| |
| return -ENOSPC; |
| } |
| |
| int sparx5_sdlb_group_get_by_index(struct sparx5 *sparx5, u32 idx, u32 *group) |
| { |
| u32 itr, next; |
| int i; |
| |
| for (i = 0; i < SPX5_SDLB_GROUP_CNT; i++) { |
| if (sparx5_sdlb_group_is_empty(sparx5, i)) |
| continue; |
| |
| itr = sparx5_sdlb_group_get_first(sparx5, i); |
| |
| for (;;) { |
| next = sparx5_sdlb_group_get_next(sparx5, i, itr); |
| |
| if (itr == idx) { |
| *group = i; |
| return 0; /* Found it */ |
| } |
| if (itr == next) |
| break; /* Was not found */ |
| |
| itr = next; |
| } |
| } |
| |
| return -EINVAL; |
| } |
| |
| static int sparx5_sdlb_group_link(struct sparx5 *sparx5, u32 group, u32 idx, |
| u32 first, u32 next, bool empty) |
| { |
| /* Stop leaking */ |
| sparx5_sdlb_group_disable(sparx5, group); |
| |
| if (empty) |
| return 0; |
| |
| /* Link insertion lb to next lb */ |
| spx5_wr(ANA_AC_SDLB_XLB_NEXT_LBSET_NEXT_SET(next) | |
| ANA_AC_SDLB_XLB_NEXT_LBGRP_SET(group), |
| sparx5, ANA_AC_SDLB_XLB_NEXT(idx)); |
| |
| /* Set the first lb */ |
| spx5_wr(ANA_AC_SDLB_XLB_START_LBSET_START_SET(first), sparx5, |
| ANA_AC_SDLB_XLB_START(group)); |
| |
| /* Start leaking */ |
| sparx5_sdlb_group_enable(sparx5, group); |
| |
| return 0; |
| }; |
| |
| int sparx5_sdlb_group_add(struct sparx5 *sparx5, u32 group, u32 idx) |
| { |
| u32 first, next; |
| |
| /* We always add to head of the list */ |
| first = idx; |
| |
| if (sparx5_sdlb_group_is_empty(sparx5, group)) |
| next = idx; |
| else |
| next = sparx5_sdlb_group_get_first(sparx5, group); |
| |
| return sparx5_sdlb_group_link(sparx5, group, idx, first, next, false); |
| } |
| |
| int sparx5_sdlb_group_del(struct sparx5 *sparx5, u32 group, u32 idx) |
| { |
| u32 first, next, prev; |
| bool empty = false; |
| |
| if (sparx5_sdlb_group_get_adjacent(sparx5, group, idx, &prev, &next, |
| &first) < 0) { |
| pr_err("%s:%d Could not find idx: %d in group: %d", __func__, |
| __LINE__, idx, group); |
| return -EINVAL; |
| } |
| |
| if (sparx5_sdlb_group_is_singular(sparx5, group)) { |
| empty = true; |
| } else if (sparx5_sdlb_group_is_last(sparx5, group, idx)) { |
| /* idx is removed, prev is now last */ |
| idx = prev; |
| next = prev; |
| } else if (sparx5_sdlb_group_is_first(sparx5, group, idx)) { |
| /* idx is removed and points to itself, first is next */ |
| first = next; |
| next = idx; |
| } else { |
| /* Next is not touched */ |
| idx = prev; |
| } |
| |
| return sparx5_sdlb_group_link(sparx5, group, idx, first, next, empty); |
| } |
| |
| void sparx5_sdlb_group_init(struct sparx5 *sparx5, u64 max_rate, u32 min_burst, |
| u32 frame_size, u32 idx) |
| { |
| u32 thres_shift, mask = 0x01, power = 0; |
| struct sparx5_sdlb_group *group; |
| u64 max_token; |
| |
| group = &sdlb_groups[idx]; |
| |
| /* Number of positions to right-shift LB's threshold value. */ |
| while ((min_burst & mask) == 0) { |
| power++; |
| mask <<= 1; |
| } |
| thres_shift = SPX5_SDLB_2CYCLES_TYPE2_THRES_OFFSET - power; |
| |
| max_token = (min_burst > SPX5_SDLB_PUP_TOKEN_MAX) ? |
| SPX5_SDLB_PUP_TOKEN_MAX : |
| min_burst; |
| group->pup_interval = |
| sparx5_sdlb_pup_interval_get(sparx5, max_token, max_rate); |
| |
| group->frame_size = frame_size; |
| |
| spx5_wr(ANA_AC_SDLB_PUP_INTERVAL_PUP_INTERVAL_SET(group->pup_interval), |
| sparx5, ANA_AC_SDLB_PUP_INTERVAL(idx)); |
| |
| spx5_wr(ANA_AC_SDLB_FRM_RATE_TOKENS_FRM_RATE_TOKENS_SET(frame_size), |
| sparx5, ANA_AC_SDLB_FRM_RATE_TOKENS(idx)); |
| |
| spx5_wr(ANA_AC_SDLB_LBGRP_MISC_THRES_SHIFT_SET(thres_shift), sparx5, |
| ANA_AC_SDLB_LBGRP_MISC(idx)); |
| } |