blob: e55fa18d67807dbfaf7ae1316ed43a7ed8a20641 [file] [log] [blame]
/*
* Copyright (c) 2012, 2016-2017, 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 isisc_init ISISC_INIT
* @{
*/
#include "sw.h"
#include "hsl.h"
#include "hsl_dev.h"
#include "hsl_port_prop.h"
#include "isisc_mib.h"
#include "isisc_port_ctrl.h"
#include "isisc_portvlan.h"
#include "isisc_vlan.h"
#include "isisc_fdb.h"
#include "isisc_qos.h"
#include "isisc_mirror.h"
#include "isisc_stp.h"
#include "isisc_rate.h"
#include "isisc_misc.h"
#include "isisc_leaky.h"
#include "isisc_igmp.h"
#include "isisc_acl.h"
#include "isisc_led.h"
#include "isisc_cosmap.h"
#include "isisc_ip.h"
#include "isisc_nat.h"
#if defined(IN_NAT_HELPER)
#include "isisc_nat_helper.h"
#endif
#include "isisc_sec.h"
#include "isisc_trunk.h"
#include "isisc_interface_ctrl.h"
#include "isisc_reg_access.h"
#include "isisc_reg.h"
#include "isisc_init.h"
#include "f1_phy.h"
static ssdk_init_cfg * isisc_cfg[SW_MAX_NR_DEV] = { 0 };
a_uint32_t isisc_nat_global_status = 0;
#if !(defined(KERNEL_MODULE) && defined(USER_MODE))
/* For isis there are five internal PHY devices and seven MAC devices.
MAC0 always connect to external MAC device.
PHY4 can connect to MAC5 or external MAC device.
MAC6 always connect to external devices.
MAC1..MAC4 connect to internal PHY0..PHY3.
*/
static sw_error_t
isisc_portproperty_init(a_uint32_t dev_id, hsl_init_mode mode)
{
hsl_port_prop_t p_type;
hsl_dev_t *pdev = NULL;
fal_port_t port_id;
pdev = hsl_dev_ptr_get(dev_id);
if (pdev == NULL)
return SW_NOT_INITIALIZED;
/* for port property set, SSDK should not generate some limitations */
for (port_id = 0; port_id < pdev->nr_ports; port_id++)
{
hsl_port_prop_portmap_set(dev_id, port_id);
for (p_type = HSL_PP_PHY; p_type < HSL_PP_BUTT; p_type++)
{
if (HSL_NO_CPU == mode)
{
SW_RTN_ON_ERROR(hsl_port_prop_set(dev_id, port_id, p_type));
continue;
}
switch (p_type)
{
case HSL_PP_PHY:
/* Only port0/port6 without PHY device */
if ((port_id != pdev->cpu_port_nr)
&& (port_id != pdev->nr_ports - 1))
{
SW_RTN_ON_ERROR(hsl_port_prop_set(dev_id, port_id, p_type));
}
break;
case HSL_PP_INCL_CPU:
/* include cpu port but exclude wan port in some cases */
/* but which port is wan port, we are no meaning */
SW_RTN_ON_ERROR(hsl_port_prop_set(dev_id, port_id, p_type));
break;
case HSL_PP_EXCL_CPU:
/* exclude cpu port and wan port in some cases */
/* which port is wan port, we are no meaning but port0 is
always CPU port */
if (port_id != pdev->cpu_port_nr)
{
SW_RTN_ON_ERROR(hsl_port_prop_set(dev_id, port_id, p_type));
}
break;
default:
break;
}
}
if (HSL_NO_CPU == mode)
{
SW_RTN_ON_ERROR(hsl_port_prop_set_phyid
(dev_id, port_id, port_id + 1));
}
else
{
if (port_id != pdev->cpu_port_nr)
{
SW_RTN_ON_ERROR(hsl_port_prop_set_phyid
(dev_id, port_id, port_id - 1));
}
}
}
return SW_OK;
}
static sw_error_t
isisc_hw_init(a_uint32_t dev_id, ssdk_init_cfg *cfg)
{
return SW_OK;
}
#endif
static sw_error_t
isisc_dev_init(a_uint32_t dev_id, hsl_init_mode cpu_mode)
{
a_uint32_t entry = 0;
sw_error_t rv;
hsl_dev_t *pdev = NULL;
pdev = hsl_dev_ptr_get(dev_id);
if (pdev == NULL)
return SW_NOT_INITIALIZED;
HSL_REG_FIELD_GET(rv, dev_id, MASK_CTL, 0, DEVICE_ID,
(a_uint8_t *) (&entry), sizeof (a_uint32_t));
SW_RTN_ON_ERROR(rv);
if (S17C_DEVICE_ID == entry)
{
pdev->nr_ports = 7;
pdev->nr_phy = 5;
pdev->cpu_port_nr = 0;
pdev->nr_vlans = 4096;
pdev->hw_vlan_query = A_TRUE;
pdev->nr_queue = 6;
pdev->cpu_mode = cpu_mode;
}
else
{
pdev->nr_ports = 6;
pdev->nr_phy = 5;
pdev->cpu_port_nr = 0;
pdev->nr_vlans = 4096;
pdev->hw_vlan_query = A_TRUE;
pdev->nr_queue = 6;
pdev->cpu_mode = cpu_mode;
}
return SW_OK;
}
static sw_error_t
_isisc_reset(a_uint32_t dev_id)
{
#if !(defined(KERNEL_MODULE) && defined(USER_MODE))
sw_error_t rv;
a_uint32_t val;
HSL_DEV_ID_CHECK(dev_id);
val = 0x1;
HSL_REG_FIELD_SET(rv, dev_id, MASK_CTL, 0, SOFT_RST,
(a_uint8_t *) (&val), sizeof (a_uint32_t));
SW_RTN_ON_ERROR(rv);
rv = isisc_hw_init(dev_id, isisc_cfg[dev_id]);
SW_RTN_ON_ERROR(rv);
ISISC_ACL_RESET(rv, dev_id);
ISISC_IP_RESET(rv, dev_id);
ISISC_NAT_RESET(rv, dev_id);
#endif
return SW_OK;
}
sw_error_t
isisc_cleanup(a_uint32_t dev_id)
{
sw_error_t rv;
if (isisc_cfg[dev_id])
{
#if defined(IN_NAT_HELPER)
if(isisc_nat_global_status) {
ISISC_NAT_HELPER_CLEANUP(rv, dev_id);
isisc_nat_global_status = 0;
}
#endif
ISISC_ACL_CLEANUP(rv, dev_id);
SW_RTN_ON_ERROR(hsl_port_prop_cleanup_by_dev(dev_id));
aos_mem_free(isisc_cfg[dev_id]);
isisc_cfg[dev_id] = NULL;
}
return SW_OK;
}
/**
* @brief reset hsl layer.
* @details Comments:
* This operation will reset hsl layer
* @param[in] dev_id device id
* @return SW_OK or error code
*/
HSL_LOCAL sw_error_t
isisc_reset(a_uint32_t dev_id)
{
sw_error_t rv;
HSL_API_LOCK;
rv = _isisc_reset(dev_id);
HSL_API_UNLOCK;
return rv;
}
/**
* @brief Init hsl layer.
* @details Comments:
* This operation will init hsl layer and hsl layer
* @param[in] dev_id device id
* @param[in] cfg configuration for initialization
* @return SW_OK or error code
*/
sw_error_t
isisc_init(a_uint32_t dev_id, ssdk_init_cfg *cfg)
{
HSL_DEV_ID_CHECK(dev_id);
if (NULL == isisc_cfg[dev_id])
{
isisc_cfg[dev_id] = aos_mem_alloc(sizeof (ssdk_init_cfg));
}
if (NULL == isisc_cfg[dev_id])
{
return SW_OUT_OF_MEM;
}
aos_mem_copy(isisc_cfg[dev_id], cfg, sizeof (ssdk_init_cfg));
SW_RTN_ON_ERROR(isisc_reg_access_init(dev_id, cfg->reg_mode));
SW_RTN_ON_ERROR(isisc_dev_init(dev_id, cfg->cpu_mode));
#if !(defined(KERNEL_MODULE) && defined(USER_MODE))
{
sw_error_t rv;
SW_RTN_ON_ERROR(hsl_port_prop_init(dev_id));
SW_RTN_ON_ERROR(hsl_port_prop_init_by_dev(dev_id));
SW_RTN_ON_ERROR(isisc_portproperty_init(dev_id, cfg->cpu_mode));
ISISC_MIB_INIT(rv, dev_id);
ISISC_PORT_CTRL_INIT(rv, dev_id);
ISISC_PORTVLAN_INIT(rv, dev_id);
ISISC_VLAN_INIT(rv, dev_id);
ISISC_FDB_INIT(rv, dev_id);
ISISC_QOS_INIT(rv, dev_id);
ISISC_STP_INIT(rv, dev_id);
ISISC_MIRR_INIT(rv, dev_id);
ISISC_RATE_INIT(rv, dev_id);
ISISC_MISC_INIT(rv, dev_id);
ISISC_LEAKY_INIT(rv, dev_id);
ISISC_IGMP_INIT(rv, dev_id);
ISISC_ACL_INIT(rv, dev_id);
ISISC_LED_INIT(rv, dev_id);
ISISC_COSMAP_INIT(rv, dev_id);
ISISC_IP_INIT(rv, dev_id);
ISISC_NAT_INIT(rv, dev_id);
ISISC_TRUNK_INIT(rv, dev_id);
ISISC_SEC_INIT(rv, dev_id);
ISISC_INTERFACE_CTRL_INIT(rv, dev_id);
{
hsl_api_t *p_api;
SW_RTN_ON_NULL(p_api = hsl_api_ptr_get(dev_id));
p_api->dev_reset = isisc_reset;
p_api->dev_clean = isisc_cleanup;
}
SW_RTN_ON_ERROR(isisc_hw_init(dev_id, cfg));
#if 0
#if defined(IN_NAT_HELPER)
if(!isisc_nat_global_status) {
ISISC_NAT_HELPER_INIT(rv, dev_id);
isisc_nat_global_status = 1;
}
#endif
#endif
}
#endif
return SW_OK;
}
/**
* @}
*/