2005-04-16 15:20:36 -07:00
/*
2005-08-10 23:03:10 -07:00
* Copyright ( c ) 2004 , 2005 Mellanox Technologies Ltd . All rights reserved .
* Copyright ( c ) 2004 , 2005 Infinicon Corporation . All rights reserved .
* Copyright ( c ) 2004 , 2005 Intel Corporation . All rights reserved .
* Copyright ( c ) 2004 , 2005 Topspin Corporation . All rights reserved .
2007-04-02 11:24:07 -04:00
* Copyright ( c ) 2004 - 2007 Voltaire Corporation . All rights reserved .
2005-08-10 23:03:10 -07:00
* Copyright ( c ) 2005 Sun Microsystems , Inc . All rights reserved .
2015-06-06 14:38:34 -04:00
* Copyright ( c ) 2014 Intel Corporation . All rights reserved .
2005-04-16 15:20:36 -07:00
*
* This software is available to you under a choice of one of two
* licenses . You may choose to be licensed under the terms of the GNU
* General Public License ( GPL ) Version 2 , available from the file
* COPYING in the main directory of this source tree , or the
* OpenIB . org BSD license below :
*
* Redistribution and use in source and binary forms , with or
* without modification , are permitted provided that the following
* conditions are met :
*
* - Redistributions of source code must retain the above
* copyright notice , this list of conditions and the following
* disclaimer .
*
* - Redistributions in binary form must reproduce the above
* copyright notice , this list of conditions and the following
* disclaimer in the documentation and / or other materials
* provided with the distribution .
*
* THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND ,
* EXPRESS OR IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY , FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT . IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
* BE LIABLE FOR ANY CLAIM , DAMAGES OR OTHER LIABILITY , WHETHER IN AN
* ACTION OF CONTRACT , TORT OR OTHERWISE , ARISING FROM , OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE .
*
*/
2005-08-25 13:40:04 -07:00
# include <rdma/ib_smi.h>
2005-04-16 15:20:36 -07:00
# include "smi.h"
2015-06-06 14:38:34 -04:00
# include "opa_smi.h"
2005-04-16 15:20:36 -07:00
2021-03-01 09:04:20 +02:00
static enum smi_action __smi_handle_dr_smp_send ( bool is_switch , u32 port_num ,
2015-06-06 14:38:25 -04:00
u8 * hop_ptr , u8 hop_cnt ,
const u8 * initial_path ,
const u8 * return_path ,
u8 direction ,
bool dr_dlid_is_permissive ,
bool dr_slid_is_permissive )
2005-04-16 15:20:36 -07:00
{
/* See section 14.2.2.2, Vol 1 IB spec */
2009-09-05 20:24:10 -07:00
/* C14-6 -- valid hop_cnt values are from 0 to 63 */
if ( hop_cnt > = IB_SMP_MAX_PATH_HOPS )
return IB_SMI_DISCARD ;
2015-06-06 14:38:25 -04:00
if ( ! direction ) {
2005-04-16 15:20:36 -07:00
/* C14-9:1 */
2015-06-06 14:38:25 -04:00
if ( hop_cnt & & * hop_ptr = = 0 ) {
( * hop_ptr ) + + ;
return ( initial_path [ * hop_ptr ] = =
2007-04-02 11:24:07 -04:00
port_num ? IB_SMI_HANDLE : IB_SMI_DISCARD ) ;
2005-04-16 15:20:36 -07:00
}
/* C14-9:2 */
2015-06-06 14:38:25 -04:00
if ( * hop_ptr & & * hop_ptr < hop_cnt ) {
2015-06-29 09:57:00 -04:00
if ( ! is_switch )
2007-04-02 11:24:07 -04:00
return IB_SMI_DISCARD ;
2005-04-16 15:20:36 -07:00
2015-06-06 14:38:25 -04:00
/* return_path set when received */
( * hop_ptr ) + + ;
return ( initial_path [ * hop_ptr ] = =
2007-04-02 11:24:07 -04:00
port_num ? IB_SMI_HANDLE : IB_SMI_DISCARD ) ;
2005-04-16 15:20:36 -07:00
}
/* C14-9:3 -- We're at the end of the DR segment of path */
2015-06-06 14:38:25 -04:00
if ( * hop_ptr = = hop_cnt ) {
/* return_path set when received */
( * hop_ptr ) + + ;
2015-06-29 09:57:00 -04:00
return ( is_switch | |
2015-06-06 14:38:25 -04:00
dr_dlid_is_permissive ?
2007-04-02 11:24:07 -04:00
IB_SMI_HANDLE : IB_SMI_DISCARD ) ;
2005-04-16 15:20:36 -07:00
}
/* C14-9:4 -- hop_ptr = hop_cnt + 1 -> give to SMA/SM */
/* C14-9:5 -- Fail unreasonable hop pointer */
2015-06-06 14:38:25 -04:00
return ( * hop_ptr = = hop_cnt + 1 ? IB_SMI_HANDLE : IB_SMI_DISCARD ) ;
2005-04-16 15:20:36 -07:00
} else {
/* C14-13:1 */
2015-06-06 14:38:25 -04:00
if ( hop_cnt & & * hop_ptr = = hop_cnt + 1 ) {
( * hop_ptr ) - - ;
return ( return_path [ * hop_ptr ] = =
2007-04-02 11:24:07 -04:00
port_num ? IB_SMI_HANDLE : IB_SMI_DISCARD ) ;
2005-04-16 15:20:36 -07:00
}
/* C14-13:2 */
2015-06-06 14:38:25 -04:00
if ( 2 < = * hop_ptr & & * hop_ptr < = hop_cnt ) {
2015-06-29 09:57:00 -04:00
if ( ! is_switch )
2007-04-02 11:24:07 -04:00
return IB_SMI_DISCARD ;
2005-04-16 15:20:36 -07:00
2015-06-06 14:38:25 -04:00
( * hop_ptr ) - - ;
return ( return_path [ * hop_ptr ] = =
2007-04-02 11:24:07 -04:00
port_num ? IB_SMI_HANDLE : IB_SMI_DISCARD ) ;
2005-04-16 15:20:36 -07:00
}
/* C14-13:3 -- at the end of the DR segment of path */
2015-06-06 14:38:25 -04:00
if ( * hop_ptr = = 1 ) {
( * hop_ptr ) - - ;
2005-04-16 15:20:36 -07:00
/* C14-13:3 -- SMPs destined for SM shouldn't be here */
2015-06-29 09:57:00 -04:00
return ( is_switch | |
2015-06-06 14:38:25 -04:00
dr_slid_is_permissive ?
2007-04-02 11:24:07 -04:00
IB_SMI_HANDLE : IB_SMI_DISCARD ) ;
2005-04-16 15:20:36 -07:00
}
/* C14-13:4 -- hop_ptr = 0 -> should have gone to SM */
2015-06-06 14:38:25 -04:00
if ( * hop_ptr = = 0 )
2007-04-02 11:24:07 -04:00
return IB_SMI_HANDLE ;
2005-04-16 15:20:36 -07:00
/* C14-13:5 -- Check for unreasonable hop pointer */
2007-04-02 11:24:07 -04:00
return IB_SMI_DISCARD ;
2005-04-16 15:20:36 -07:00
}
}
2015-06-06 14:38:25 -04:00
/*
* Fixup a directed route SMP for sending
* Return IB_SMI_DISCARD if the SMP should be discarded
*/
enum smi_action smi_handle_dr_smp_send ( struct ib_smp * smp ,
2021-03-01 09:04:20 +02:00
bool is_switch , u32 port_num )
2015-06-06 14:38:25 -04:00
{
2015-06-29 09:57:00 -04:00
return __smi_handle_dr_smp_send ( is_switch , port_num ,
2015-06-06 14:38:25 -04:00
& smp - > hop_ptr , smp - > hop_cnt ,
smp - > initial_path ,
smp - > return_path ,
ib_get_smp_direction ( smp ) ,
smp - > dr_dlid = = IB_LID_PERMISSIVE ,
smp - > dr_slid = = IB_LID_PERMISSIVE ) ;
}
2015-06-06 14:38:34 -04:00
enum smi_action opa_smi_handle_dr_smp_send ( struct opa_smp * smp ,
2021-03-01 09:04:20 +02:00
bool is_switch , u32 port_num )
2015-06-06 14:38:34 -04:00
{
2015-06-29 09:57:00 -04:00
return __smi_handle_dr_smp_send ( is_switch , port_num ,
2015-06-06 14:38:34 -04:00
& smp - > hop_ptr , smp - > hop_cnt ,
smp - > route . dr . initial_path ,
smp - > route . dr . return_path ,
opa_get_smp_direction ( smp ) ,
smp - > route . dr . dr_dlid = =
OPA_LID_PERMISSIVE ,
smp - > route . dr . dr_slid = =
OPA_LID_PERMISSIVE ) ;
}
2021-03-01 09:04:20 +02:00
static enum smi_action __smi_handle_dr_smp_recv ( bool is_switch , u32 port_num ,
2015-06-06 14:38:26 -04:00
int phys_port_cnt ,
u8 * hop_ptr , u8 hop_cnt ,
const u8 * initial_path ,
u8 * return_path ,
u8 direction ,
bool dr_dlid_is_permissive ,
bool dr_slid_is_permissive )
2005-04-16 15:20:36 -07:00
{
/* See section 14.2.2.2, Vol 1 IB spec */
2009-09-05 20:24:10 -07:00
/* C14-6 -- valid hop_cnt values are from 0 to 63 */
if ( hop_cnt > = IB_SMP_MAX_PATH_HOPS )
return IB_SMI_DISCARD ;
2015-06-06 14:38:26 -04:00
if ( ! direction ) {
2005-04-16 15:20:36 -07:00
/* C14-9:1 -- sender should have incremented hop_ptr */
2015-06-06 14:38:26 -04:00
if ( hop_cnt & & * hop_ptr = = 0 )
2007-04-02 11:24:07 -04:00
return IB_SMI_DISCARD ;
2005-04-16 15:20:36 -07:00
/* C14-9:2 -- intermediate hop */
2015-06-06 14:38:26 -04:00
if ( * hop_ptr & & * hop_ptr < hop_cnt ) {
2015-06-29 09:57:00 -04:00
if ( ! is_switch )
2007-04-02 11:24:07 -04:00
return IB_SMI_DISCARD ;
2005-04-16 15:20:36 -07:00
2015-06-06 14:38:26 -04:00
return_path [ * hop_ptr ] = port_num ;
/* hop_ptr updated when sending */
return ( initial_path [ * hop_ptr + 1 ] < = phys_port_cnt ?
2007-04-02 11:24:07 -04:00
IB_SMI_HANDLE : IB_SMI_DISCARD ) ;
2005-04-16 15:20:36 -07:00
}
/* C14-9:3 -- We're at the end of the DR segment of path */
2015-06-06 14:38:26 -04:00
if ( * hop_ptr = = hop_cnt ) {
2005-04-16 15:20:36 -07:00
if ( hop_cnt )
2015-06-06 14:38:26 -04:00
return_path [ * hop_ptr ] = port_num ;
/* hop_ptr updated when sending */
2005-04-16 15:20:36 -07:00
2015-06-29 09:57:00 -04:00
return ( is_switch | |
2015-06-06 14:38:26 -04:00
dr_dlid_is_permissive ?
2007-04-02 11:24:07 -04:00
IB_SMI_HANDLE : IB_SMI_DISCARD ) ;
2005-04-16 15:20:36 -07:00
}
/* C14-9:4 -- hop_ptr = hop_cnt + 1 -> give to SMA/SM */
/* C14-9:5 -- fail unreasonable hop pointer */
2015-06-06 14:38:26 -04:00
return ( * hop_ptr = = hop_cnt + 1 ? IB_SMI_HANDLE : IB_SMI_DISCARD ) ;
2005-04-16 15:20:36 -07:00
} else {
/* C14-13:1 */
2015-06-06 14:38:26 -04:00
if ( hop_cnt & & * hop_ptr = = hop_cnt + 1 ) {
( * hop_ptr ) - - ;
return ( return_path [ * hop_ptr ] = =
2007-04-02 11:24:07 -04:00
port_num ? IB_SMI_HANDLE : IB_SMI_DISCARD ) ;
2005-04-16 15:20:36 -07:00
}
/* C14-13:2 */
2015-06-06 14:38:26 -04:00
if ( 2 < = * hop_ptr & & * hop_ptr < = hop_cnt ) {
2015-06-29 09:57:00 -04:00
if ( ! is_switch )
2007-04-02 11:24:07 -04:00
return IB_SMI_DISCARD ;
2005-04-16 15:20:36 -07:00
2015-06-06 14:38:26 -04:00
/* hop_ptr updated when sending */
return ( return_path [ * hop_ptr - 1 ] < = phys_port_cnt ?
2007-04-02 11:24:07 -04:00
IB_SMI_HANDLE : IB_SMI_DISCARD ) ;
2005-04-16 15:20:36 -07:00
}
/* C14-13:3 -- We're at the end of the DR segment of path */
2015-06-06 14:38:26 -04:00
if ( * hop_ptr = = 1 ) {
if ( dr_slid_is_permissive ) {
2005-04-16 15:20:36 -07:00
/* giving SMP to SM - update hop_ptr */
2015-06-06 14:38:26 -04:00
( * hop_ptr ) - - ;
2007-04-02 11:24:07 -04:00
return IB_SMI_HANDLE ;
2005-04-16 15:20:36 -07:00
}
2015-06-06 14:38:26 -04:00
/* hop_ptr updated when sending */
2015-06-29 09:57:00 -04:00
return ( is_switch ? IB_SMI_HANDLE : IB_SMI_DISCARD ) ;
2005-04-16 15:20:36 -07:00
}
/* C14-13:4 -- hop_ptr = 0 -> give to SM */
/* C14-13:5 -- Check for unreasonable hop pointer */
2015-06-06 14:38:26 -04:00
return ( * hop_ptr = = 0 ? IB_SMI_HANDLE : IB_SMI_DISCARD ) ;
2005-04-16 15:20:36 -07:00
}
}
2015-06-06 14:38:26 -04:00
/*
* Adjust information for a received SMP
* Return IB_SMI_DISCARD if the SMP should be dropped
*/
2015-06-29 09:57:00 -04:00
enum smi_action smi_handle_dr_smp_recv ( struct ib_smp * smp , bool is_switch ,
2021-03-01 09:04:20 +02:00
u32 port_num , int phys_port_cnt )
2015-06-06 14:38:26 -04:00
{
2015-06-29 09:57:00 -04:00
return __smi_handle_dr_smp_recv ( is_switch , port_num , phys_port_cnt ,
2015-06-06 14:38:26 -04:00
& smp - > hop_ptr , smp - > hop_cnt ,
smp - > initial_path ,
smp - > return_path ,
ib_get_smp_direction ( smp ) ,
smp - > dr_dlid = = IB_LID_PERMISSIVE ,
smp - > dr_slid = = IB_LID_PERMISSIVE ) ;
}
2015-06-06 14:38:34 -04:00
/*
* Adjust information for a received SMP
* Return IB_SMI_DISCARD if the SMP should be dropped
*/
2015-06-29 09:57:00 -04:00
enum smi_action opa_smi_handle_dr_smp_recv ( struct opa_smp * smp , bool is_switch ,
2021-03-01 09:04:20 +02:00
u32 port_num , int phys_port_cnt )
2015-06-06 14:38:34 -04:00
{
2015-06-29 09:57:00 -04:00
return __smi_handle_dr_smp_recv ( is_switch , port_num , phys_port_cnt ,
2015-06-06 14:38:34 -04:00
& smp - > hop_ptr , smp - > hop_cnt ,
smp - > route . dr . initial_path ,
smp - > route . dr . return_path ,
opa_get_smp_direction ( smp ) ,
smp - > route . dr . dr_dlid = =
OPA_LID_PERMISSIVE ,
smp - > route . dr . dr_slid = =
OPA_LID_PERMISSIVE ) ;
}
2015-06-06 14:38:27 -04:00
static enum smi_forward_action __smi_check_forward_dr_smp ( u8 hop_ptr , u8 hop_cnt ,
u8 direction ,
bool dr_dlid_is_permissive ,
bool dr_slid_is_permissive )
2005-04-16 15:20:36 -07:00
{
2015-06-06 14:38:27 -04:00
if ( ! direction ) {
2005-04-16 15:20:36 -07:00
/* C14-9:2 -- intermediate hop */
if ( hop_ptr & & hop_ptr < hop_cnt )
2007-05-14 17:21:52 -04:00
return IB_SMI_FORWARD ;
2005-04-16 15:20:36 -07:00
/* C14-9:3 -- at the end of the DR segment of path */
if ( hop_ptr = = hop_cnt )
2015-06-06 14:38:27 -04:00
return ( dr_dlid_is_permissive ?
2007-04-02 11:24:07 -04:00
IB_SMI_SEND : IB_SMI_LOCAL ) ;
2005-04-16 15:20:36 -07:00
/* C14-9:4 -- hop_ptr = hop_cnt + 1 -> give to SMA/SM */
if ( hop_ptr = = hop_cnt + 1 )
2007-04-02 11:24:07 -04:00
return IB_SMI_SEND ;
2005-04-16 15:20:36 -07:00
} else {
2007-04-02 11:24:07 -04:00
/* C14-13:2 -- intermediate hop */
2005-04-16 15:20:36 -07:00
if ( 2 < = hop_ptr & & hop_ptr < = hop_cnt )
2007-05-14 17:21:52 -04:00
return IB_SMI_FORWARD ;
2005-04-16 15:20:36 -07:00
/* C14-13:3 -- at the end of the DR segment of path */
if ( hop_ptr = = 1 )
2015-06-06 14:38:27 -04:00
return ( ! dr_slid_is_permissive ?
2007-04-02 11:24:07 -04:00
IB_SMI_SEND : IB_SMI_LOCAL ) ;
2005-04-16 15:20:36 -07:00
}
2007-04-02 11:24:07 -04:00
return IB_SMI_LOCAL ;
2015-06-06 14:38:27 -04:00
}
enum smi_forward_action smi_check_forward_dr_smp ( struct ib_smp * smp )
{
return __smi_check_forward_dr_smp ( smp - > hop_ptr , smp - > hop_cnt ,
ib_get_smp_direction ( smp ) ,
smp - > dr_dlid = = IB_LID_PERMISSIVE ,
smp - > dr_slid = = IB_LID_PERMISSIVE ) ;
2005-04-16 15:20:36 -07:00
}
2007-05-14 17:21:52 -04:00
2015-06-06 14:38:34 -04:00
enum smi_forward_action opa_smi_check_forward_dr_smp ( struct opa_smp * smp )
{
return __smi_check_forward_dr_smp ( smp - > hop_ptr , smp - > hop_cnt ,
opa_get_smp_direction ( smp ) ,
smp - > route . dr . dr_dlid = =
OPA_LID_PERMISSIVE ,
smp - > route . dr . dr_slid = =
OPA_LID_PERMISSIVE ) ;
}
2007-05-14 17:21:52 -04:00
/*
* Return the forwarding port number from initial_path for outgoing SMP and
* from return_path for returning SMP
*/
int smi_get_fwd_port ( struct ib_smp * smp )
{
return ( ! ib_get_smp_direction ( smp ) ? smp - > initial_path [ smp - > hop_ptr + 1 ] :
smp - > return_path [ smp - > hop_ptr - 1 ] ) ;
}
2015-06-06 14:38:34 -04:00
/*
* Return the forwarding port number from initial_path for outgoing SMP and
* from return_path for returning SMP
*/
int opa_smi_get_fwd_port ( struct opa_smp * smp )
{
return ! opa_get_smp_direction ( smp ) ? smp - > route . dr . initial_path [ smp - > hop_ptr + 1 ] :
smp - > route . dr . return_path [ smp - > hop_ptr - 1 ] ;
}