|  | // 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)); | 
|  | } |