/* | |
* Copyright (c) 2014, The Linux Foundation. All rights reserved. | |
* Permission to use, copy, modify, and/or distribute this software for | |
* any purpose with or without fee is hereby granted, provided that the | |
* above copyright notice and this permission notice appear in all copies. | |
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES | |
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF | |
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR | |
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES | |
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN | |
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT | |
* OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. | |
*/ | |
/** | |
* @defgroup dess_vlan DESS_VLAN | |
* @{ | |
*/ | |
#include "sw.h" | |
#include "hsl.h" | |
#include "hsl_dev.h" | |
#include "hsl_port_prop.h" | |
#include "dess_vlan.h" | |
#include "dess_reg.h" | |
#define MAX_VLAN_ID 4095 | |
#define VLAN_FLUSH 1 | |
#define VLAN_LOAD_ENTRY 2 | |
#define VLAN_PURGE_ENTRY 3 | |
#define VLAN_REMOVE_PORT 4 | |
#define VLAN_NEXT_ENTRY 5 | |
#define VLAN_FIND_ENTRY 6 | |
static void | |
_dess_vlan_hw_to_sw(a_uint32_t reg[], fal_vlan_t * vlan_entry) | |
{ | |
a_uint32_t i, data, tmp; | |
aos_mem_zero(vlan_entry, sizeof (fal_vlan_t)); | |
SW_GET_FIELD_BY_REG(VLAN_TABLE_FUNC1, VLAN_ID, data, reg[1]); | |
vlan_entry->vid = data & 0xfff; | |
SW_GET_FIELD_BY_REG(VLAN_TABLE_FUNC0, IVL_EN, data, reg[0]); | |
if (1 == data) | |
{ | |
vlan_entry->fid = vlan_entry->vid; | |
} | |
else | |
{ | |
vlan_entry->fid = FAL_SVL_FID; | |
} | |
SW_GET_FIELD_BY_REG(VLAN_TABLE_FUNC0, LEARN_DIS, data, reg[0]); | |
if (1 == data) | |
{ | |
vlan_entry->learn_dis = A_TRUE; | |
} | |
else | |
{ | |
vlan_entry->learn_dis = A_FALSE; | |
} | |
SW_GET_FIELD_BY_REG(VLAN_TABLE_FUNC0, VT_PRI_EN, data, reg[0]); | |
if (1 == data) | |
{ | |
vlan_entry->vid_pri_en = A_TRUE; | |
SW_GET_FIELD_BY_REG(VLAN_TABLE_FUNC0, VT_PRI, data, reg[0]); | |
vlan_entry->vid_pri = data & 0xff; | |
} | |
else | |
{ | |
vlan_entry->vid_pri_en = A_FALSE; | |
} | |
SW_GET_FIELD_BY_REG(VLAN_TABLE_FUNC0, VID_MEM, data, reg[0]); | |
for (i = 0; i < 7; i++) | |
{ | |
tmp = (data >> (i << 1)) & 0x3UL; | |
if (0 == tmp) | |
{ | |
vlan_entry->mem_ports |= (0x1UL << i); | |
vlan_entry->unmodify_ports |= (0x1UL << i); | |
} | |
else if (1 == tmp) | |
{ | |
vlan_entry->mem_ports |= (0x1UL << i); | |
vlan_entry->untagged_ports |= (0x1UL << i); | |
} | |
else if (2 == tmp) | |
{ | |
vlan_entry->mem_ports |= (0x1UL << i); | |
vlan_entry->tagged_ports |= (0x1UL << i); | |
} | |
} | |
return; | |
} | |
static sw_error_t | |
_dess_vlan_sw_to_hw(a_uint32_t dev_id, const fal_vlan_t * vlan_entry, | |
a_uint32_t reg[]) | |
{ | |
a_uint32_t i, tag, untag, unmodify, member = 0; | |
if (vlan_entry->vid > MAX_VLAN_ID) | |
{ | |
return SW_OUT_OF_RANGE; | |
} | |
if (A_FALSE == | |
hsl_mports_prop_check(dev_id, vlan_entry->mem_ports, HSL_PP_INCL_CPU)) | |
{ | |
return SW_BAD_PARAM; | |
} | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC0, VT_VALID, 1, reg[0]); | |
if (FAL_SVL_FID == vlan_entry->fid) | |
{ | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC0, IVL_EN, 0, reg[0]); | |
} | |
else if (vlan_entry->vid == vlan_entry->fid) | |
{ | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC0, IVL_EN, 1, reg[0]); | |
} | |
else | |
{ | |
return SW_BAD_VALUE; | |
} | |
if (A_TRUE == vlan_entry->learn_dis) | |
{ | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC0, LEARN_DIS, 1, reg[0]); | |
} | |
else | |
{ | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC0, LEARN_DIS, 0, reg[0]); | |
} | |
for (i = 0; i < 7; i++) | |
{ | |
if ((vlan_entry->mem_ports >> i) & 0x1UL) | |
{ | |
tag = (vlan_entry->tagged_ports >> i) & 0x1UL; | |
untag = (vlan_entry->untagged_ports >> i) & 0x1UL; | |
unmodify = (vlan_entry->unmodify_ports >> i) & 0x1UL; | |
if ((0 == (tag + untag + unmodify)) | |
|| (1 < (tag + untag + unmodify))) | |
{ | |
return SW_BAD_VALUE; | |
} | |
if (tag) | |
{ | |
member |= (2 << (i << 1)); | |
} | |
else if (untag) | |
{ | |
member |= (1 << (i << 1)); | |
} | |
} | |
else | |
{ | |
member |= (3 << (i << 1)); | |
} | |
} | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC0, VID_MEM, member, reg[0]); | |
if (A_TRUE == vlan_entry->vid_pri_en) | |
{ | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC0, VT_PRI_EN, 1, reg[0]); | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC0, VT_PRI, vlan_entry->vid_pri, | |
reg[0]); | |
} | |
else | |
{ | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC0, VT_PRI_EN, 0, reg[0]); | |
} | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC1, VLAN_ID, vlan_entry->vid, reg[1]); | |
return SW_OK; | |
} | |
static sw_error_t | |
_dess_vlan_down_to_hw(a_uint32_t dev_id, a_uint32_t reg[]) | |
{ | |
sw_error_t rv; | |
HSL_REG_ENTRY_SET(rv, dev_id, VLAN_TABLE_FUNC0, 0, | |
(a_uint8_t *) (®[0]), sizeof (a_uint32_t)); | |
SW_RTN_ON_ERROR(rv); | |
HSL_REG_ENTRY_SET(rv, dev_id, VLAN_TABLE_FUNC1, 0, | |
(a_uint8_t *) (®[1]), sizeof (a_uint32_t)); | |
return rv; | |
} | |
static sw_error_t | |
_dess_vlan_up_to_sw(a_uint32_t dev_id, a_uint32_t reg[]) | |
{ | |
sw_error_t rv; | |
HSL_REG_ENTRY_GET(rv, dev_id, VLAN_TABLE_FUNC0, 0, | |
(a_uint8_t *) (®[0]), sizeof (a_uint32_t)); | |
SW_RTN_ON_ERROR(rv); | |
HSL_REG_ENTRY_GET(rv, dev_id, VLAN_TABLE_FUNC1, 0, | |
(a_uint8_t *) (®[1]), sizeof (a_uint32_t)); | |
return rv; | |
} | |
static sw_error_t | |
_dess_vlan_commit(a_uint32_t dev_id, a_uint32_t op) | |
{ | |
a_uint32_t vt_busy = 1, i = 0x1000, vt_full, val; | |
sw_error_t rv; | |
while (vt_busy && --i) | |
{ | |
HSL_REG_FIELD_GET(rv, dev_id, VLAN_TABLE_FUNC1, 0, VT_BUSY, | |
(a_uint8_t *) (&vt_busy), sizeof (a_uint32_t)); | |
SW_RTN_ON_ERROR(rv); | |
aos_udelay(5); | |
} | |
if (i == 0) | |
return SW_BUSY; | |
HSL_REG_ENTRY_GET(rv, dev_id, VLAN_TABLE_FUNC1, 0, | |
(a_uint8_t *) (&val), sizeof (a_uint32_t)); | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC1, VT_FUNC, op, val); | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC1, VT_BUSY, 1, val); | |
HSL_REG_ENTRY_SET(rv, dev_id, VLAN_TABLE_FUNC1, 0, | |
(a_uint8_t *) (&val), sizeof (a_uint32_t)); | |
SW_RTN_ON_ERROR(rv); | |
vt_busy = 1; | |
i = 0x1000; | |
while (vt_busy && --i) | |
{ | |
HSL_REG_FIELD_GET(rv, dev_id, VLAN_TABLE_FUNC1, 0, VT_BUSY, | |
(a_uint8_t *) (&vt_busy), sizeof (a_uint32_t)); | |
SW_RTN_ON_ERROR(rv); | |
aos_udelay(5); | |
} | |
if (i == 0) | |
return SW_FAIL; | |
HSL_REG_FIELD_GET(rv, dev_id, VLAN_TABLE_FUNC1, 0, VT_FULL_VIO, | |
(a_uint8_t *) (&vt_full), sizeof (a_uint32_t)); | |
SW_RTN_ON_ERROR(rv); | |
if (vt_full) | |
{ | |
val = 0x10; | |
HSL_REG_ENTRY_SET(rv, dev_id, VLAN_TABLE_FUNC1, 0, | |
(a_uint8_t *) (&val), sizeof (a_uint32_t)); | |
SW_RTN_ON_ERROR(rv); | |
if (VLAN_LOAD_ENTRY == op) | |
{ | |
return SW_FULL; | |
} | |
else if (VLAN_PURGE_ENTRY == op) | |
{ | |
return SW_NOT_FOUND; | |
} | |
} | |
HSL_REG_FIELD_GET(rv, dev_id, VLAN_TABLE_FUNC0, 0, VT_VALID, | |
(a_uint8_t *) (&val), sizeof (a_uint32_t)); | |
SW_RTN_ON_ERROR(rv); | |
if (!val) | |
{ | |
if (VLAN_FIND_ENTRY == op) | |
return SW_NOT_FOUND; | |
if (VLAN_NEXT_ENTRY == op) | |
return SW_NO_MORE; | |
} | |
return SW_OK; | |
} | |
static sw_error_t | |
_dess_vlan_hwentry_get(a_uint32_t dev_id, a_uint32_t vlan_id, a_uint32_t reg[]) | |
{ | |
sw_error_t rv; | |
if (vlan_id > MAX_VLAN_ID) | |
{ | |
return SW_OUT_OF_RANGE; | |
} | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC1, VLAN_ID, vlan_id, reg[1]); | |
HSL_REG_ENTRY_SET(rv, dev_id, VLAN_TABLE_FUNC1, 0, | |
(a_uint8_t *) (®[1]), sizeof (a_uint32_t)); | |
SW_RTN_ON_ERROR(rv); | |
rv = _dess_vlan_commit(dev_id, VLAN_FIND_ENTRY); | |
SW_RTN_ON_ERROR(rv); | |
rv = _dess_vlan_up_to_sw(dev_id, reg); | |
return rv; | |
} | |
static sw_error_t | |
_dess_vlan_entry_append(a_uint32_t dev_id, const fal_vlan_t * vlan_entry) | |
{ | |
sw_error_t rv; | |
a_uint32_t reg[2] = { 0 }; | |
HSL_DEV_ID_CHECK(dev_id); | |
rv = _dess_vlan_sw_to_hw(dev_id, vlan_entry, reg); | |
SW_RTN_ON_ERROR(rv); | |
rv = _dess_vlan_down_to_hw(dev_id, reg); | |
SW_RTN_ON_ERROR(rv); | |
rv = _dess_vlan_commit(dev_id, VLAN_LOAD_ENTRY); | |
return rv; | |
} | |
static sw_error_t | |
_dess_vlan_create(a_uint32_t dev_id, a_uint32_t vlan_id) | |
{ | |
sw_error_t rv; | |
a_uint32_t reg[2] = { 0 }; | |
HSL_DEV_ID_CHECK(dev_id); | |
if (vlan_id > MAX_VLAN_ID) | |
{ | |
return SW_OUT_OF_RANGE; | |
} | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC0, VT_VALID, 1, reg[0]); | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC0, IVL_EN, 1, reg[0]); | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC0, LEARN_DIS, 0, reg[0]); | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC0, VID_MEM, 0x3fff, reg[0]); | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC1, VLAN_ID, vlan_id, reg[1]); | |
rv = _dess_vlan_down_to_hw(dev_id, reg); | |
SW_RTN_ON_ERROR(rv); | |
rv = _dess_vlan_commit(dev_id, VLAN_LOAD_ENTRY); | |
return rv; | |
} | |
static sw_error_t | |
_dess_vlan_next(a_uint32_t dev_id, a_uint32_t vlan_id, fal_vlan_t * p_vlan) | |
{ | |
sw_error_t rv; | |
a_uint32_t reg[2] = { 0 }; | |
HSL_DEV_ID_CHECK(dev_id); | |
if (FAL_NEXT_ENTRY_FIRST_ID == vlan_id) | |
{ | |
rv = _dess_vlan_hwentry_get(dev_id, 0, reg); | |
if (SW_OK == rv) | |
{ | |
_dess_vlan_hw_to_sw(reg, p_vlan); | |
return SW_OK; | |
} | |
else | |
{ | |
vlan_id = 0; | |
} | |
} | |
if (vlan_id > MAX_VLAN_ID) | |
return SW_OUT_OF_RANGE; | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC1, VLAN_ID, vlan_id, reg[1]); | |
HSL_REG_ENTRY_SET(rv, dev_id, VLAN_TABLE_FUNC1, 0, | |
(a_uint8_t *) (®[1]), sizeof (a_uint32_t)); | |
SW_RTN_ON_ERROR(rv); | |
rv = _dess_vlan_commit(dev_id, VLAN_NEXT_ENTRY); | |
SW_RTN_ON_ERROR(rv); | |
rv = _dess_vlan_up_to_sw(dev_id, reg); | |
SW_RTN_ON_ERROR(rv); | |
_dess_vlan_hw_to_sw(reg, p_vlan); | |
if (0 == p_vlan->vid) | |
{ | |
return SW_NO_MORE; | |
} | |
else | |
{ | |
return SW_OK; | |
} | |
} | |
static sw_error_t | |
_dess_vlan_find(a_uint32_t dev_id, a_uint32_t vlan_id, fal_vlan_t * p_vlan) | |
{ | |
sw_error_t rv; | |
a_uint32_t reg[2] = { 0 }; | |
HSL_DEV_ID_CHECK(dev_id); | |
rv = _dess_vlan_hwentry_get(dev_id, vlan_id, reg); | |
SW_RTN_ON_ERROR(rv); | |
_dess_vlan_hw_to_sw(reg, p_vlan); | |
return SW_OK; | |
} | |
static sw_error_t | |
_dess_vlan_delete(a_uint32_t dev_id, a_uint32_t vlan_id) | |
{ | |
sw_error_t rv; | |
a_uint32_t reg; | |
HSL_DEV_ID_CHECK(dev_id); | |
if (vlan_id > MAX_VLAN_ID) | |
{ | |
return SW_OUT_OF_RANGE; | |
} | |
reg = (a_int32_t) vlan_id; | |
HSL_REG_FIELD_SET(rv, dev_id, VLAN_TABLE_FUNC1, 0, VLAN_ID, | |
(a_uint8_t *) (®), sizeof (a_uint32_t)); | |
SW_RTN_ON_ERROR(rv); | |
rv = _dess_vlan_commit(dev_id, VLAN_PURGE_ENTRY); | |
return rv; | |
} | |
static sw_error_t | |
_dess_vlan_flush(a_uint32_t dev_id) | |
{ | |
sw_error_t rv; | |
HSL_DEV_ID_CHECK(dev_id); | |
rv = _dess_vlan_commit(dev_id, VLAN_FLUSH); | |
return rv; | |
} | |
static sw_error_t | |
_dess_vlan_fid_set(a_uint32_t dev_id, a_uint32_t vlan_id, a_uint32_t fid) | |
{ | |
sw_error_t rv; | |
a_uint32_t reg[2] = { 0 }; | |
HSL_DEV_ID_CHECK(dev_id); | |
if ((MAX_VLAN_ID < fid) && (FAL_SVL_FID != fid)) | |
{ | |
return SW_BAD_PARAM; | |
} | |
if ((MAX_VLAN_ID >= fid) && (vlan_id != fid)) | |
{ | |
return SW_BAD_PARAM; | |
} | |
rv = _dess_vlan_hwentry_get(dev_id, vlan_id, reg); | |
SW_RTN_ON_ERROR(rv); | |
if (FAL_SVL_FID == fid) | |
{ | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC0, IVL_EN, 0, reg[0]); | |
} | |
else | |
{ | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC0, IVL_EN, 1, reg[0]); | |
} | |
rv = _dess_vlan_down_to_hw(dev_id, reg); | |
SW_RTN_ON_ERROR(rv); | |
rv = _dess_vlan_commit(dev_id, VLAN_LOAD_ENTRY); | |
if (SW_FULL == rv) | |
{ | |
rv = SW_OK; | |
} | |
return rv; | |
} | |
static sw_error_t | |
_dess_vlan_fid_get(a_uint32_t dev_id, a_uint32_t vlan_id, a_uint32_t * fid) | |
{ | |
sw_error_t rv; | |
a_uint32_t data, reg[2] = { 0 }; | |
HSL_DEV_ID_CHECK(dev_id); | |
rv = _dess_vlan_hwentry_get(dev_id, vlan_id, reg); | |
SW_RTN_ON_ERROR(rv); | |
SW_GET_FIELD_BY_REG(VLAN_TABLE_FUNC0, IVL_EN, data, reg[0]); | |
if (data) | |
{ | |
*fid = vlan_id; | |
} | |
else | |
{ | |
*fid = FAL_SVL_FID; | |
} | |
return SW_OK; | |
} | |
static sw_error_t | |
_dess_vlan_member_update(a_uint32_t dev_id, a_uint32_t vlan_id, | |
fal_port_t port_id, a_uint32_t port_info) | |
{ | |
sw_error_t rv; | |
a_uint32_t data, reg[2] = { 0 }; | |
HSL_DEV_ID_CHECK(dev_id); | |
if (A_TRUE != hsl_port_prop_check(dev_id, port_id, HSL_PP_INCL_CPU)) | |
{ | |
return SW_BAD_PARAM; | |
} | |
rv = _dess_vlan_hwentry_get(dev_id, vlan_id, reg); | |
SW_RTN_ON_ERROR(rv); | |
SW_GET_FIELD_BY_REG(VLAN_TABLE_FUNC0, VID_MEM, data, reg[0]); | |
data &= (~(0x3 << (port_id << 1))); | |
data |= ((port_info & 0x3) << (port_id << 1)); | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC0, VID_MEM, data, reg[0]); | |
rv = _dess_vlan_down_to_hw(dev_id, reg); | |
SW_RTN_ON_ERROR(rv); | |
rv = _dess_vlan_commit(dev_id, VLAN_LOAD_ENTRY); | |
if (SW_FULL == rv) | |
{ | |
rv = SW_OK; | |
} | |
return rv; | |
} | |
static sw_error_t | |
_dess_vlan_member_add(a_uint32_t dev_id, a_uint32_t vlan_id, | |
fal_port_t port_id, fal_pt_1q_egmode_t port_info) | |
{ | |
sw_error_t rv; | |
a_uint32_t info = 0; | |
if (FAL_EG_UNMODIFIED == port_info) | |
{ | |
info = 0; | |
} | |
else if (FAL_EG_TAGGED == port_info) | |
{ | |
info = 0x2; | |
} | |
else if (FAL_EG_UNTAGGED == port_info) | |
{ | |
info = 0x1; | |
} | |
else | |
{ | |
return SW_BAD_PARAM; | |
} | |
rv = _dess_vlan_member_update(dev_id, vlan_id, port_id, info); | |
return rv; | |
} | |
static sw_error_t | |
_dess_vlan_member_del(a_uint32_t dev_id, a_uint32_t vlan_id, fal_port_t port_id) | |
{ | |
sw_error_t rv; | |
a_uint32_t info = 0x3; | |
rv = _dess_vlan_member_update(dev_id, vlan_id, port_id, info); | |
return rv; | |
} | |
static sw_error_t | |
_dess_vlan_learning_state_set(a_uint32_t dev_id, a_uint32_t vlan_id, | |
a_bool_t enable) | |
{ | |
sw_error_t rv; | |
a_uint32_t reg[2] = { 0 }; | |
HSL_DEV_ID_CHECK(dev_id); | |
rv = _dess_vlan_hwentry_get(dev_id, vlan_id, reg); | |
SW_RTN_ON_ERROR(rv); | |
if (A_TRUE == enable) | |
{ | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC0, LEARN_DIS, 0, reg[0]); | |
} | |
else if (A_FALSE == enable) | |
{ | |
SW_SET_REG_BY_FIELD(VLAN_TABLE_FUNC0, LEARN_DIS, 1, reg[0]); | |
} | |
else | |
{ | |
return SW_BAD_PARAM; | |
} | |
rv = _dess_vlan_down_to_hw(dev_id, reg); | |
SW_RTN_ON_ERROR(rv); | |
rv = _dess_vlan_commit(dev_id, VLAN_LOAD_ENTRY); | |
if (SW_FULL == rv) | |
{ | |
rv = SW_OK; | |
} | |
return rv; | |
} | |
static sw_error_t | |
_dess_vlan_learning_state_get(a_uint32_t dev_id, a_uint32_t vlan_id, | |
a_bool_t * enable) | |
{ | |
sw_error_t rv; | |
a_uint32_t data, reg[2] = { 0 }; | |
HSL_DEV_ID_CHECK(dev_id); | |
rv = _dess_vlan_hwentry_get(dev_id, vlan_id, reg); | |
SW_RTN_ON_ERROR(rv); | |
SW_GET_FIELD_BY_REG(VLAN_TABLE_FUNC0, LEARN_DIS, data, reg[0]); | |
if (data) | |
{ | |
*enable = A_FALSE; | |
} | |
else | |
{ | |
*enable = A_TRUE; | |
} | |
return SW_OK; | |
} | |
/** | |
* @brief Append a vlan entry on paticular device. | |
* @param[in] dev_id device id | |
* @param[in] vlan_entry vlan entry | |
* @return SW_OK or error code | |
*/ | |
HSL_LOCAL sw_error_t | |
dess_vlan_entry_append(a_uint32_t dev_id, const fal_vlan_t * vlan_entry) | |
{ | |
sw_error_t rv; | |
HSL_API_LOCK; | |
rv = _dess_vlan_entry_append(dev_id, vlan_entry); | |
HSL_API_UNLOCK; | |
return rv; | |
} | |
/** | |
* @brief Creat a vlan entry through vlan id on a paticular device. | |
* @details Comments: | |
* After this operation the member ports of the created vlan entry are null. | |
* @param[in] dev_id device id | |
* @param[in] vlan_id vlan id | |
* @return SW_OK or error code | |
*/ | |
HSL_LOCAL sw_error_t | |
dess_vlan_create(a_uint32_t dev_id, a_uint32_t vlan_id) | |
{ | |
sw_error_t rv; | |
HSL_API_LOCK; | |
rv = _dess_vlan_create(dev_id, vlan_id); | |
HSL_API_UNLOCK; | |
return rv; | |
} | |
/** | |
* @brief Next a vlan entry through vlan id on a paticular device. | |
* @details Comments: | |
* If the value of vid is zero this operation will get the first entry. | |
* @param[in] dev_id device id | |
* @param[in] vlan_id vlan id | |
* @param[out] p_vlan vlan entry | |
* @return SW_OK or error code | |
*/ | |
HSL_LOCAL sw_error_t | |
dess_vlan_next(a_uint32_t dev_id, a_uint32_t vlan_id, fal_vlan_t * p_vlan) | |
{ | |
sw_error_t rv; | |
HSL_API_LOCK; | |
rv = _dess_vlan_next(dev_id, vlan_id, p_vlan); | |
HSL_API_UNLOCK; | |
return rv; | |
} | |
/** | |
* @brief Find a vlan entry through vlan id on paticular device. | |
* @param[in] dev_id device id | |
* @param[in] vlan_id vlan id | |
* @param[out] p_vlan vlan entry | |
* @return SW_OK or error code | |
*/ | |
HSL_LOCAL sw_error_t | |
dess_vlan_find(a_uint32_t dev_id, a_uint32_t vlan_id, fal_vlan_t * p_vlan) | |
{ | |
sw_error_t rv; | |
HSL_API_LOCK; | |
rv = _dess_vlan_find(dev_id, vlan_id, p_vlan); | |
HSL_API_UNLOCK; | |
return rv; | |
} | |
/** | |
* @brief Delete a vlan entry through vlan id on a paticular device. | |
* @param[in] dev_id device id | |
* @param[in] vlan_id vlan id | |
* @return SW_OK or error code | |
*/ | |
HSL_LOCAL sw_error_t | |
dess_vlan_delete(a_uint32_t dev_id, a_uint32_t vlan_id) | |
{ | |
sw_error_t rv; | |
HSL_API_LOCK; | |
rv = _dess_vlan_delete(dev_id, vlan_id); | |
HSL_API_UNLOCK; | |
return rv; | |
} | |
/** | |
* @brief Flush all vlan entries on a paticular device. | |
* @param[in] dev_id device id | |
* @return SW_OK or error code | |
*/ | |
HSL_LOCAL sw_error_t | |
dess_vlan_flush(a_uint32_t dev_id) | |
{ | |
sw_error_t rv; | |
HSL_API_LOCK; | |
rv = _dess_vlan_flush(dev_id); | |
HSL_API_UNLOCK; | |
return rv; | |
} | |
/** | |
* @brief Set FID of a paticular vlan entry on a paticular device. | |
* @param[in] dev_id device id | |
* @param[in] vlan_id vlan id | |
* @param[in] fid FDB id | |
* @return SW_OK or error code | |
*/ | |
HSL_LOCAL sw_error_t | |
dess_vlan_fid_set(a_uint32_t dev_id, a_uint32_t vlan_id, a_uint32_t fid) | |
{ | |
sw_error_t rv; | |
HSL_API_LOCK; | |
rv = _dess_vlan_fid_set(dev_id, vlan_id, fid); | |
HSL_API_UNLOCK; | |
return rv; | |
} | |
/** | |
* @brief Get FID of a paticular vlan entry on a paticular device. | |
* @param[in] dev_id device id | |
* @param[in] vlan_id vlan id | |
* @param[out] fid FDB id | |
* @return SW_OK or error code | |
*/ | |
HSL_LOCAL sw_error_t | |
dess_vlan_fid_get(a_uint32_t dev_id, a_uint32_t vlan_id, a_uint32_t * fid) | |
{ | |
sw_error_t rv; | |
HSL_API_LOCK; | |
rv = _dess_vlan_fid_get(dev_id, vlan_id, fid); | |
HSL_API_UNLOCK; | |
return rv; | |
} | |
/** | |
* @brief Add a port member to a paticular vlan entry on a paticular device. | |
* @param[in] dev_id device id | |
* @param[in] vlan_id vlan id | |
* @param[in] port_id port id | |
* @param[in] port_info port tag information | |
* @return SW_OK or error code | |
*/ | |
HSL_LOCAL sw_error_t | |
dess_vlan_member_add(a_uint32_t dev_id, a_uint32_t vlan_id, | |
fal_port_t port_id, fal_pt_1q_egmode_t port_info) | |
{ | |
sw_error_t rv; | |
HSL_API_LOCK; | |
rv = _dess_vlan_member_add(dev_id, vlan_id, port_id, port_info); | |
HSL_API_UNLOCK; | |
return rv; | |
} | |
/** | |
* @brief Del a port member from a paticular vlan entry on a paticular device. | |
* @param[in] dev_id device id | |
* @param[in] vlan_id vlan id | |
* @param[in] port_id port id | |
* @return SW_OK or error code | |
*/ | |
HSL_LOCAL sw_error_t | |
dess_vlan_member_del(a_uint32_t dev_id, a_uint32_t vlan_id, fal_port_t port_id) | |
{ | |
sw_error_t rv; | |
HSL_API_LOCK; | |
rv = _dess_vlan_member_del(dev_id, vlan_id, port_id); | |
HSL_API_UNLOCK; | |
return rv; | |
} | |
/** | |
* @brief Set FDB learning status of a paticular vlan entry on a paticular device. | |
* @param[in] dev_id device id | |
* @param[in] vlan_id vlan id | |
* @param[in] enable A_TRUE or A_FALSE | |
* @return SW_OK or error code | |
*/ | |
HSL_LOCAL sw_error_t | |
dess_vlan_learning_state_set(a_uint32_t dev_id, a_uint32_t vlan_id, | |
a_bool_t enable) | |
{ | |
sw_error_t rv; | |
HSL_API_LOCK; | |
rv = _dess_vlan_learning_state_set(dev_id, vlan_id, enable); | |
HSL_API_UNLOCK; | |
return rv; | |
} | |
/** | |
* @brief Get FDB learning status of a paticular vlan entry on a paticular device. | |
* @param[in] dev_id device id | |
* @param[in] vlan_id vlan id | |
* @param[out] enable A_TRUE or A_FALSE | |
* @return SW_OK or error code | |
*/ | |
HSL_LOCAL sw_error_t | |
dess_vlan_learning_state_get(a_uint32_t dev_id, a_uint32_t vlan_id, | |
a_bool_t * enable) | |
{ | |
sw_error_t rv; | |
HSL_API_LOCK; | |
rv = _dess_vlan_learning_state_get(dev_id, vlan_id, enable); | |
HSL_API_UNLOCK; | |
return rv; | |
} | |
sw_error_t | |
dess_vlan_init(a_uint32_t dev_id) | |
{ | |
hsl_api_t *p_api; | |
HSL_DEV_ID_CHECK(dev_id); | |
#ifndef HSL_STANDALONG | |
SW_RTN_ON_NULL(p_api = hsl_api_ptr_get(dev_id)); | |
p_api->vlan_entry_append = dess_vlan_entry_append; | |
p_api->vlan_creat = dess_vlan_create; | |
p_api->vlan_delete = dess_vlan_delete; | |
p_api->vlan_next = dess_vlan_next; | |
p_api->vlan_find = dess_vlan_find; | |
p_api->vlan_flush = dess_vlan_flush; | |
p_api->vlan_fid_set = dess_vlan_fid_set; | |
p_api->vlan_fid_get = dess_vlan_fid_get; | |
p_api->vlan_member_add = dess_vlan_member_add; | |
p_api->vlan_member_del = dess_vlan_member_del; | |
p_api->vlan_learning_state_set = dess_vlan_learning_state_set; | |
p_api->vlan_learning_state_get = dess_vlan_learning_state_get; | |
#endif | |
return SW_OK; | |
} | |
/** | |
* @} | |
*/ | |