mirror of https://github.com/F-Stack/f-stack.git
1703 lines
67 KiB
C
1703 lines
67 KiB
C
|
/* Copyright (c) 2008-2011 Freescale Semiconductor, Inc.
|
||
|
* All rights reserved.
|
||
|
*
|
||
|
* 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.
|
||
|
* * Neither the name of Freescale Semiconductor nor the
|
||
|
* names of its contributors may be used to endorse or promote products
|
||
|
* derived from this software without specific prior written permission.
|
||
|
*
|
||
|
*
|
||
|
* ALTERNATIVELY, this software may be distributed under the terms of the
|
||
|
* GNU General Public License ("GPL") as published by the Free Software
|
||
|
* Foundation, either version 2 of that License or (at your option) any
|
||
|
* later version.
|
||
|
*
|
||
|
* THIS SOFTWARE IS PROVIDED BY Freescale Semiconductor ``AS IS'' AND ANY
|
||
|
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||
|
* DISCLAIMED. IN NO EVENT SHALL Freescale Semiconductor BE LIABLE FOR ANY
|
||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||
|
*/
|
||
|
|
||
|
/******************************************************************************
|
||
|
@File fm_plcr.c
|
||
|
|
||
|
@Description FM PCD POLICER...
|
||
|
*//***************************************************************************/
|
||
|
#include "std_ext.h"
|
||
|
#include "error_ext.h"
|
||
|
#include "string_ext.h"
|
||
|
#include "debug_ext.h"
|
||
|
#include "net_ext.h"
|
||
|
#include "fm_ext.h"
|
||
|
|
||
|
#include "fm_common.h"
|
||
|
#include "fm_pcd.h"
|
||
|
#include "fm_hc.h"
|
||
|
#include "fm_pcd_ipc.h"
|
||
|
|
||
|
|
||
|
static bool FmPcdPlcrIsProfileShared(t_Handle h_FmPcd, uint16_t absoluteProfileId)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
uint16_t i;
|
||
|
|
||
|
SANITY_CHECK_RETURN_VALUE(p_FmPcd, E_INVALID_HANDLE, FALSE);
|
||
|
|
||
|
for(i=0;i<p_FmPcd->p_FmPcdPlcr->numOfSharedProfiles;i++)
|
||
|
if(p_FmPcd->p_FmPcdPlcr->sharedProfilesIds[i] == absoluteProfileId)
|
||
|
return TRUE;
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
static t_Error SetProfileNia(t_FmPcd *p_FmPcd, e_FmPcdEngine nextEngine, u_FmPcdPlcrNextEngineParams *p_NextEngineParams, uint32_t *nextAction)
|
||
|
{
|
||
|
uint32_t nia;
|
||
|
uint16_t absoluteProfileId = (uint16_t)(PTR_TO_UINT(p_NextEngineParams->h_Profile)-1);
|
||
|
uint8_t relativeSchemeId, physicatSchemeId;
|
||
|
|
||
|
nia = FM_PCD_PLCR_NIA_VALID;
|
||
|
|
||
|
switch (nextEngine)
|
||
|
{
|
||
|
case e_FM_PCD_DONE :
|
||
|
switch (p_NextEngineParams->action)
|
||
|
{
|
||
|
case e_FM_PCD_DROP_FRAME :
|
||
|
nia |= (NIA_ENG_BMI | NIA_BMI_AC_DISCARD);
|
||
|
break;
|
||
|
case e_FM_PCD_ENQ_FRAME:
|
||
|
nia |= (NIA_ENG_BMI | NIA_BMI_AC_ENQ_FRAME);
|
||
|
break;
|
||
|
default:
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_SELECTION, NO_MSG);
|
||
|
}
|
||
|
break;
|
||
|
case e_FM_PCD_KG:
|
||
|
physicatSchemeId = (uint8_t)(PTR_TO_UINT(p_NextEngineParams->h_DirectScheme)-1);
|
||
|
relativeSchemeId = FmPcdKgGetRelativeSchemeId(p_FmPcd, physicatSchemeId);
|
||
|
if(relativeSchemeId == FM_PCD_KG_NUM_OF_SCHEMES)
|
||
|
RETURN_ERROR(MAJOR, E_NOT_IN_RANGE, NO_MSG);
|
||
|
if (!FmPcdKgIsSchemeValidSw(p_FmPcd, relativeSchemeId))
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_STATE, ("Invalid direct scheme."));
|
||
|
if(!KgIsSchemeAlwaysDirect(p_FmPcd, relativeSchemeId))
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_STATE, ("Policer Profile may point only to a scheme that is always direct."));
|
||
|
nia |= NIA_ENG_KG | NIA_KG_DIRECT | physicatSchemeId;
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR:
|
||
|
if(!FmPcdPlcrIsProfileShared(p_FmPcd, absoluteProfileId))
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_STATE, ("Next profile must be a shared profile"));
|
||
|
if(!FmPcdPlcrIsProfileValid(p_FmPcd, absoluteProfileId))
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_STATE, ("Invalid profile "));
|
||
|
nia |= NIA_ENG_PLCR | NIA_PLCR_ABSOLUTE | absoluteProfileId;
|
||
|
break;
|
||
|
default:
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_SELECTION, NO_MSG);
|
||
|
}
|
||
|
|
||
|
*nextAction = nia;
|
||
|
|
||
|
return E_OK;
|
||
|
}
|
||
|
|
||
|
static uint32_t FPP_Function(uint32_t fpp)
|
||
|
{
|
||
|
if(fpp > 15)
|
||
|
return 15 - (0x1f - fpp);
|
||
|
else
|
||
|
return 16 + fpp;
|
||
|
}
|
||
|
|
||
|
static void GetInfoRateReg(e_FmPcdPlcrRateMode rateMode,
|
||
|
uint32_t rate,
|
||
|
uint64_t tsuInTenthNano,
|
||
|
uint32_t fppShift,
|
||
|
uint64_t *p_Integer,
|
||
|
uint64_t *p_Fraction)
|
||
|
{
|
||
|
uint64_t tmp, div;
|
||
|
|
||
|
if(rateMode == e_FM_PCD_PLCR_BYTE_MODE)
|
||
|
{
|
||
|
/* now we calculate the initial integer for the bigger rate */
|
||
|
/* from Kbps to Bytes/TSU */
|
||
|
tmp = (uint64_t)rate;
|
||
|
tmp *= 1000; /* kb --> b */
|
||
|
tmp *= tsuInTenthNano; /* bps --> bpTsu(in 10nano) */
|
||
|
|
||
|
div = 1000000000; /* nano */
|
||
|
div *= 10; /* 10 nano */
|
||
|
div *= 8; /* bit to byte */
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
/* now we calculate the initial integer for the bigger rate */
|
||
|
/* from Kbps to Bytes/TSU */
|
||
|
tmp = (uint64_t)rate;
|
||
|
tmp *= tsuInTenthNano; /* bps --> bpTsu(in 10nano) */
|
||
|
|
||
|
div = 1000000000; /* nano */
|
||
|
div *= 10; /* 10 nano */
|
||
|
}
|
||
|
*p_Integer = (tmp<<fppShift)/div;
|
||
|
|
||
|
/* for calculating the fraction, we will recalculate cir and deduct the integer.
|
||
|
* For precision, we will multiply by 2^16. we do not divid back, since we write
|
||
|
* this value as fraction - see spec.
|
||
|
*/
|
||
|
*p_Fraction = (((tmp<<fppShift)<<16) - ((*p_Integer<<16)*div))/div;
|
||
|
}
|
||
|
|
||
|
/* .......... */
|
||
|
|
||
|
static void calcRates(t_Handle h_FmPcd, t_FmPcdPlcrNonPassthroughAlgParams *p_NonPassthroughAlgParam,
|
||
|
uint32_t *cir, uint32_t *cbs, uint32_t *pir_eir, uint32_t *pbs_ebs, uint32_t *fpp)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
uint64_t integer, fraction;
|
||
|
uint32_t temp, tsuInTenthNanos, bitFor1Micro;
|
||
|
uint8_t fppShift=0;
|
||
|
|
||
|
bitFor1Micro = FmGetTimeStampScale(p_FmPcd->h_Fm); /* TimeStamp per nano seconds units */
|
||
|
/* we want the tsu to count 10 nano for better precision normally tsu is 3.9 nano, now we will get 39 */
|
||
|
tsuInTenthNanos = (uint32_t)(1000*10/(1<<bitFor1Micro));
|
||
|
|
||
|
/* we choose the faster rate to calibrate fpp */
|
||
|
if (p_NonPassthroughAlgParam->comittedInfoRate > p_NonPassthroughAlgParam->peakOrAccessiveInfoRate)
|
||
|
GetInfoRateReg(p_NonPassthroughAlgParam->rateMode, p_NonPassthroughAlgParam->comittedInfoRate, tsuInTenthNanos, 0, &integer, &fraction);
|
||
|
else
|
||
|
GetInfoRateReg(p_NonPassthroughAlgParam->rateMode, p_NonPassthroughAlgParam->peakOrAccessiveInfoRate, tsuInTenthNanos, 0, &integer, &fraction);
|
||
|
|
||
|
|
||
|
/* we shift integer, as in cir/pir it is represented by the MSB 16 bits, and
|
||
|
* the LSB bits are for the fraction */
|
||
|
temp = (uint32_t)((integer<<16) & 0x00000000FFFFFFFF);
|
||
|
/* temp is effected by the rate. For low rates it may be as low as 0, and then we'll
|
||
|
* take max fpp=31.
|
||
|
* For high rates it will never exceed the 32 bit reg (after the 16 shift), as it is
|
||
|
* limited by the 10G physical port.
|
||
|
*/
|
||
|
if(temp != 0)
|
||
|
{
|
||
|
/* count zeroes left of the higher used bit (in order to shift the value such that
|
||
|
* unused bits may be used for fraction).
|
||
|
*/
|
||
|
while ((temp & 0x80000000) == 0)
|
||
|
{
|
||
|
temp = temp << 1;
|
||
|
fppShift++;
|
||
|
}
|
||
|
if(fppShift > 15)
|
||
|
{
|
||
|
REPORT_ERROR(MAJOR, E_INVALID_SELECTION, ("timeStampPeriod to Information rate ratio is too small"));
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
temp = (uint32_t)fraction; /* fraction will alyas be smaller than 2^16 */
|
||
|
if(!temp)
|
||
|
/* integer and fraction are 0, we set fpp to its max val */
|
||
|
fppShift = 31;
|
||
|
else
|
||
|
{
|
||
|
/* integer was 0 but fraction is not. fpp is 16 for the integer,
|
||
|
* + all left zeroes of the fraction. */
|
||
|
fppShift=16;
|
||
|
/* count zeroes left of the higher used bit (in order to shift the value such that
|
||
|
* unused bits may be used for fraction).
|
||
|
*/
|
||
|
while ((temp & 0x8000) == 0)
|
||
|
{
|
||
|
temp = temp << 1;
|
||
|
fppShift++;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* This means that the FM TS register will now be used so that 'count' bits are for
|
||
|
* fraction and the rest for integer */
|
||
|
/* now we re-calculate cir */
|
||
|
GetInfoRateReg(p_NonPassthroughAlgParam->rateMode, p_NonPassthroughAlgParam->comittedInfoRate, tsuInTenthNanos, fppShift, &integer, &fraction);
|
||
|
*cir = (uint32_t)(integer << 16 | (fraction & 0xFFFF));
|
||
|
GetInfoRateReg(p_NonPassthroughAlgParam->rateMode, p_NonPassthroughAlgParam->peakOrAccessiveInfoRate, tsuInTenthNanos, fppShift, &integer, &fraction);
|
||
|
*pir_eir = (uint32_t)(integer << 16 | (fraction & 0xFFFF));
|
||
|
|
||
|
*cbs = p_NonPassthroughAlgParam->comittedBurstSize;
|
||
|
*pbs_ebs = p_NonPassthroughAlgParam->peakOrAccessiveBurstSize;
|
||
|
|
||
|
/* get fpp as it should be written to reg.*/
|
||
|
*fpp = FPP_Function(fppShift);
|
||
|
|
||
|
}
|
||
|
|
||
|
static void WritePar(t_FmPcd *p_FmPcd, uint32_t par)
|
||
|
{
|
||
|
t_FmPcdPlcrRegs *p_FmPcdPlcrRegs = p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs;
|
||
|
|
||
|
ASSERT_COND(FmIsMaster(p_FmPcd->h_Fm));
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->fmpl_par, par);
|
||
|
|
||
|
while(GET_UINT32(p_FmPcdPlcrRegs->fmpl_par) & FM_PCD_PLCR_PAR_GO) ;
|
||
|
|
||
|
}
|
||
|
|
||
|
/*********************************************/
|
||
|
/*............Policer Exception..............*/
|
||
|
/*********************************************/
|
||
|
static void PcdPlcrException(t_Handle h_FmPcd)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd *)h_FmPcd;
|
||
|
uint32_t event, mask, force;
|
||
|
|
||
|
ASSERT_COND(FmIsMaster(p_FmPcd->h_Fm));
|
||
|
event = GET_UINT32(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs->fmpl_evr);
|
||
|
mask = GET_UINT32(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs->fmpl_ier);
|
||
|
|
||
|
event &= mask;
|
||
|
|
||
|
/* clear the forced events */
|
||
|
force = GET_UINT32(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs->fmpl_ifr);
|
||
|
if(force & event)
|
||
|
WRITE_UINT32(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs->fmpl_ifr, force & ~event);
|
||
|
|
||
|
|
||
|
WRITE_UINT32(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs->fmpl_evr, event);
|
||
|
|
||
|
if(event & FM_PCD_PLCR_PRAM_SELF_INIT_COMPLETE)
|
||
|
p_FmPcd->f_Exception(p_FmPcd->h_App,e_FM_PCD_PLCR_EXCEPTION_PRAM_SELF_INIT_COMPLETE);
|
||
|
if(event & FM_PCD_PLCR_ATOMIC_ACTION_COMPLETE)
|
||
|
p_FmPcd->f_Exception(p_FmPcd->h_App,e_FM_PCD_PLCR_EXCEPTION_ATOMIC_ACTION_COMPLETE);
|
||
|
|
||
|
}
|
||
|
|
||
|
/* ..... */
|
||
|
|
||
|
static void PcdPlcrErrorException(t_Handle h_FmPcd)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd *)h_FmPcd;
|
||
|
uint32_t event, force, captureReg, mask;
|
||
|
|
||
|
ASSERT_COND(FmIsMaster(p_FmPcd->h_Fm));
|
||
|
event = GET_UINT32(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs->fmpl_eevr);
|
||
|
mask = GET_UINT32(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs->fmpl_eier);
|
||
|
|
||
|
event &= mask;
|
||
|
|
||
|
/* clear the forced events */
|
||
|
force = GET_UINT32(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs->fmpl_eifr);
|
||
|
if(force & event)
|
||
|
WRITE_UINT32(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs->fmpl_eifr, force & ~event);
|
||
|
|
||
|
WRITE_UINT32(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs->fmpl_eevr, event);
|
||
|
|
||
|
if(event & FM_PCD_PLCR_DOUBLE_ECC)
|
||
|
p_FmPcd->f_Exception(p_FmPcd->h_App,e_FM_PCD_PLCR_EXCEPTION_DOUBLE_ECC);
|
||
|
if(event & FM_PCD_PLCR_INIT_ENTRY_ERROR)
|
||
|
{
|
||
|
captureReg = GET_UINT32(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs->fmpl_upcr);
|
||
|
/*ASSERT_COND(captureReg & PLCR_ERR_UNINIT_CAP);
|
||
|
p_UnInitCapt->profileNum = (uint8_t)(captureReg & PLCR_ERR_UNINIT_NUM_MASK);
|
||
|
p_UnInitCapt->portId = (uint8_t)((captureReg & PLCR_ERR_UNINIT_PID_MASK) >>PLCR_ERR_UNINIT_PID_SHIFT) ;
|
||
|
p_UnInitCapt->absolute = (bool)(captureReg & PLCR_ERR_UNINIT_ABSOLUTE_MASK);*/
|
||
|
p_FmPcd->f_FmPcdIndexedException(p_FmPcd->h_App,e_FM_PCD_PLCR_EXCEPTION_INIT_ENTRY_ERROR,(uint16_t)(captureReg & PLCR_ERR_UNINIT_NUM_MASK));
|
||
|
WRITE_UINT32(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs->fmpl_upcr, PLCR_ERR_UNINIT_CAP);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void FmPcdPlcrUpatePointedOwner(t_Handle h_FmPcd, uint16_t absoluteProfileId, bool add)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
|
||
|
ASSERT_COND(p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].valid);
|
||
|
|
||
|
if(add)
|
||
|
p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].pointedOwners++;
|
||
|
else
|
||
|
p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].pointedOwners--;
|
||
|
}
|
||
|
|
||
|
uint32_t FmPcdPlcrGetPointedOwners(t_Handle h_FmPcd, uint16_t absoluteProfileId)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
|
||
|
ASSERT_COND(p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].valid);
|
||
|
|
||
|
return p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].pointedOwners;
|
||
|
}
|
||
|
uint32_t FmPcdPlcrGetRequiredAction(t_Handle h_FmPcd, uint16_t absoluteProfileId)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
|
||
|
ASSERT_COND(p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].valid);
|
||
|
|
||
|
return p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].requiredAction;
|
||
|
}
|
||
|
|
||
|
t_Error FmPcdPlcrAllocProfiles(t_Handle h_FmPcd, uint8_t hardwarePortId, uint16_t numOfProfiles)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
t_FmPcdIpcPlcrAllocParams ipcPlcrParams;
|
||
|
t_Error err = E_OK;
|
||
|
uint16_t base;
|
||
|
uint16_t swPortIndex = 0;
|
||
|
t_FmPcdIpcMsg msg;
|
||
|
uint32_t replyLength;
|
||
|
t_FmPcdIpcReply reply;
|
||
|
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd, E_INVALID_HANDLE);
|
||
|
|
||
|
if(!numOfProfiles)
|
||
|
return E_OK;
|
||
|
|
||
|
memset(&ipcPlcrParams, 0, sizeof(ipcPlcrParams));
|
||
|
|
||
|
if(p_FmPcd->guestId != NCSW_MASTER_ID)
|
||
|
{
|
||
|
/* Alloc resources using IPC messaging */
|
||
|
memset(&reply, 0, sizeof(reply));
|
||
|
memset(&msg, 0, sizeof(msg));
|
||
|
ipcPlcrParams.num = numOfProfiles;
|
||
|
ipcPlcrParams.hardwarePortId = hardwarePortId;
|
||
|
msg.msgId = FM_PCD_ALLOC_PROFILES;
|
||
|
memcpy(msg.msgBody, &ipcPlcrParams, sizeof(ipcPlcrParams));
|
||
|
replyLength = sizeof(uint32_t) + sizeof(uint16_t);
|
||
|
if ((err = XX_IpcSendMessage(p_FmPcd->h_IpcSession,
|
||
|
(uint8_t*)&msg,
|
||
|
sizeof(msg.msgId) +sizeof(ipcPlcrParams),
|
||
|
(uint8_t*)&reply,
|
||
|
&replyLength,
|
||
|
NULL,
|
||
|
NULL)) != E_OK)
|
||
|
RETURN_ERROR(MAJOR, err,NO_MSG);
|
||
|
if (replyLength != sizeof(uint32_t) + sizeof(uint16_t))
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_VALUE, ("IPC reply length mismatch"));
|
||
|
if((t_Error)reply.error != E_OK)
|
||
|
RETURN_ERROR(MAJOR, (t_Error)reply.error, ("PLCR profiles allocation failed"));
|
||
|
|
||
|
memcpy((uint8_t*)&base, reply.replyBody, sizeof(uint16_t));
|
||
|
}
|
||
|
else /* master */
|
||
|
{
|
||
|
err = PlcrAllocProfiles(p_FmPcd, hardwarePortId, numOfProfiles, &base);
|
||
|
if(err)
|
||
|
RETURN_ERROR(MAJOR, err,NO_MSG);
|
||
|
}
|
||
|
HW_PORT_ID_TO_SW_PORT_INDX(swPortIndex, hardwarePortId);
|
||
|
|
||
|
p_FmPcd->p_FmPcdPlcr->portsMapping[swPortIndex].numOfProfiles = numOfProfiles;
|
||
|
p_FmPcd->p_FmPcdPlcr->portsMapping[swPortIndex].profilesBase = base;
|
||
|
|
||
|
return E_OK;
|
||
|
}
|
||
|
|
||
|
t_Error FmPcdPlcrFreeProfiles(t_Handle h_FmPcd, uint8_t hardwarePortId)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
t_FmPcdIpcPlcrAllocParams ipcPlcrParams;
|
||
|
t_Error err = E_OK;
|
||
|
uint16_t swPortIndex = 0;
|
||
|
t_FmPcdIpcMsg msg;
|
||
|
uint32_t replyLength;
|
||
|
t_FmPcdIpcReply reply;
|
||
|
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd, E_INVALID_HANDLE);
|
||
|
|
||
|
HW_PORT_ID_TO_SW_PORT_INDX(swPortIndex, hardwarePortId);
|
||
|
|
||
|
if(p_FmPcd->guestId != NCSW_MASTER_ID)
|
||
|
{
|
||
|
/* Alloc resources using IPC messaging */
|
||
|
memset(&reply, 0, sizeof(reply));
|
||
|
memset(&msg, 0, sizeof(msg));
|
||
|
ipcPlcrParams.num = p_FmPcd->p_FmPcdPlcr->portsMapping[swPortIndex].numOfProfiles;
|
||
|
ipcPlcrParams.hardwarePortId = hardwarePortId;
|
||
|
ipcPlcrParams.plcrProfilesBase = p_FmPcd->p_FmPcdPlcr->portsMapping[swPortIndex].profilesBase;
|
||
|
msg.msgId = FM_PCD_FREE_PROFILES;
|
||
|
memcpy(msg.msgBody, &ipcPlcrParams, sizeof(ipcPlcrParams));
|
||
|
replyLength = sizeof(uint32_t);
|
||
|
if ((err = XX_IpcSendMessage(p_FmPcd->h_IpcSession,
|
||
|
(uint8_t*)&msg,
|
||
|
sizeof(msg.msgId) +sizeof(ipcPlcrParams),
|
||
|
(uint8_t*)&reply,
|
||
|
&replyLength,
|
||
|
NULL,
|
||
|
NULL)) != E_OK)
|
||
|
RETURN_ERROR(MAJOR, err,NO_MSG);
|
||
|
if (replyLength != sizeof(uint32_t))
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_VALUE, ("IPC reply length mismatch"));
|
||
|
if ((t_Error)reply.error != E_OK)
|
||
|
RETURN_ERROR(MINOR, (t_Error)reply.error, ("PLCR Free Profiles failed"));
|
||
|
}
|
||
|
else /* master */
|
||
|
{
|
||
|
err = PlcrFreeProfiles(p_FmPcd, hardwarePortId, p_FmPcd->p_FmPcdPlcr->portsMapping[swPortIndex].numOfProfiles, p_FmPcd->p_FmPcdPlcr->portsMapping[swPortIndex].profilesBase);
|
||
|
if(err)
|
||
|
RETURN_ERROR(MAJOR, err,NO_MSG);
|
||
|
}
|
||
|
p_FmPcd->p_FmPcdPlcr->portsMapping[swPortIndex].numOfProfiles = 0;
|
||
|
p_FmPcd->p_FmPcdPlcr->portsMapping[swPortIndex].profilesBase = 0;
|
||
|
|
||
|
return E_OK;
|
||
|
}
|
||
|
|
||
|
bool FmPcdPlcrIsProfileValid(t_Handle h_FmPcd, uint16_t absoluteProfileId)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
t_FmPcdPlcr *p_FmPcdPlcr = p_FmPcd->p_FmPcdPlcr;
|
||
|
|
||
|
return p_FmPcdPlcr->profiles[absoluteProfileId].valid;
|
||
|
}
|
||
|
|
||
|
t_Error PlcrAllocProfiles(t_FmPcd *p_FmPcd, uint8_t hardwarePortId, uint16_t numOfProfiles, uint16_t *p_Base)
|
||
|
{
|
||
|
t_FmPcdPlcrRegs *p_Regs = p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs;
|
||
|
uint32_t profilesFound, log2Num, tmpReg32;
|
||
|
uint32_t intFlags;
|
||
|
uint16_t first, i;
|
||
|
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd, E_INVALID_HANDLE);
|
||
|
|
||
|
ASSERT_COND(FmIsMaster(p_FmPcd->h_Fm));
|
||
|
if(!numOfProfiles)
|
||
|
return E_OK;
|
||
|
|
||
|
ASSERT_COND(hardwarePortId);
|
||
|
|
||
|
if (numOfProfiles>FM_PCD_PLCR_NUM_ENTRIES)
|
||
|
RETURN_ERROR(MINOR, E_INVALID_VALUE, ("numProfiles is too big."));
|
||
|
|
||
|
if (!POWER_OF_2(numOfProfiles))
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_VALUE, ("numProfiles must be a power of 2."));
|
||
|
|
||
|
intFlags = FmPcdLock(p_FmPcd);
|
||
|
|
||
|
if(GET_UINT32(p_Regs->fmpl_pmr[hardwarePortId-1]) & FM_PCD_PLCR_PMR_V)
|
||
|
{
|
||
|
FmPcdUnlock(p_FmPcd, intFlags);
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_VALUE, ("The requesting port has already an allocated profiles window."));
|
||
|
}
|
||
|
|
||
|
first = 0;
|
||
|
profilesFound = 0;
|
||
|
for(i=0;i<FM_PCD_PLCR_NUM_ENTRIES;)
|
||
|
{
|
||
|
if(!p_FmPcd->p_FmPcdPlcr->profiles[i].profilesMng.allocated)
|
||
|
{
|
||
|
profilesFound++;
|
||
|
i++;
|
||
|
if(profilesFound == numOfProfiles)
|
||
|
break;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
profilesFound = 0;
|
||
|
/* advance i to the next aligned address */
|
||
|
first = i = (uint8_t)(first + numOfProfiles);
|
||
|
}
|
||
|
}
|
||
|
if(profilesFound == numOfProfiles)
|
||
|
{
|
||
|
for(i = first; i<first + numOfProfiles; i++)
|
||
|
{
|
||
|
p_FmPcd->p_FmPcdPlcr->profiles[i].profilesMng.allocated = TRUE;
|
||
|
p_FmPcd->p_FmPcdPlcr->profiles[i].profilesMng.ownerId = hardwarePortId;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
FmPcdUnlock(p_FmPcd, intFlags);
|
||
|
RETURN_ERROR(MINOR, E_FULL, ("No profiles."));
|
||
|
}
|
||
|
|
||
|
/**********************FMPL_PMRx******************/
|
||
|
LOG2((uint64_t)numOfProfiles, log2Num);
|
||
|
tmpReg32 = first;
|
||
|
tmpReg32 |= log2Num << 16;
|
||
|
tmpReg32 |= FM_PCD_PLCR_PMR_V;
|
||
|
WRITE_UINT32(p_Regs->fmpl_pmr[hardwarePortId-1], tmpReg32);
|
||
|
|
||
|
*p_Base = first;
|
||
|
|
||
|
FmPcdUnlock(p_FmPcd, intFlags);
|
||
|
|
||
|
return E_OK;
|
||
|
}
|
||
|
|
||
|
t_Error PlcrAllocSharedProfiles(t_FmPcd *p_FmPcd, uint16_t numOfProfiles, uint16_t *profilesIds)
|
||
|
{
|
||
|
uint32_t profilesFound;
|
||
|
uint16_t i, k=0;
|
||
|
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd, E_INVALID_HANDLE);
|
||
|
|
||
|
ASSERT_COND(FmIsMaster(p_FmPcd->h_Fm));
|
||
|
if(!numOfProfiles)
|
||
|
return E_OK;
|
||
|
|
||
|
if (numOfProfiles>FM_PCD_PLCR_NUM_ENTRIES)
|
||
|
RETURN_ERROR(MINOR, E_INVALID_VALUE, ("numProfiles is too big."));
|
||
|
|
||
|
profilesFound = 0;
|
||
|
for(i=0;i<FM_PCD_PLCR_NUM_ENTRIES; i++)
|
||
|
{
|
||
|
if(!p_FmPcd->p_FmPcdPlcr->profiles[i].profilesMng.allocated)
|
||
|
{
|
||
|
profilesFound++;
|
||
|
profilesIds[k] = i;
|
||
|
k++;
|
||
|
if(profilesFound == numOfProfiles)
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if(profilesFound != numOfProfiles)
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_STATE,NO_MSG);
|
||
|
for(i = 0;i<k;i++)
|
||
|
{
|
||
|
p_FmPcd->p_FmPcdPlcr->profiles[profilesIds[i]].profilesMng.allocated = TRUE;
|
||
|
p_FmPcd->p_FmPcdPlcr->profiles[profilesIds[i]].profilesMng.ownerId = 0;
|
||
|
}
|
||
|
|
||
|
return E_OK;
|
||
|
}
|
||
|
|
||
|
t_Error PlcrFreeProfiles(t_FmPcd *p_FmPcd, uint8_t hardwarePortId, uint16_t numOfProfiles, uint16_t base)
|
||
|
{
|
||
|
t_FmPcdPlcrRegs *p_Regs = p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs;
|
||
|
uint16_t i;
|
||
|
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd, E_INVALID_HANDLE);
|
||
|
SANITY_CHECK_RETURN_ERROR(!p_FmPcd->p_FmPcdDriverParam, E_INVALID_HANDLE);
|
||
|
|
||
|
ASSERT_COND(FmIsMaster(p_FmPcd->h_Fm));
|
||
|
ASSERT_COND(IN_RANGE(1, hardwarePortId, 63));
|
||
|
WRITE_UINT32(p_Regs->fmpl_pmr[hardwarePortId-1], 0);
|
||
|
|
||
|
for(i = base; i<base+numOfProfiles;i++)
|
||
|
{
|
||
|
ASSERT_COND(p_FmPcd->p_FmPcdPlcr->profiles[i].profilesMng.ownerId == hardwarePortId);
|
||
|
ASSERT_COND(p_FmPcd->p_FmPcdPlcr->profiles[i].profilesMng.allocated);
|
||
|
|
||
|
p_FmPcd->p_FmPcdPlcr->profiles[i].profilesMng.allocated = FALSE;
|
||
|
p_FmPcd->p_FmPcdPlcr->profiles[i].profilesMng.ownerId = 0;
|
||
|
}
|
||
|
|
||
|
return E_OK;
|
||
|
}
|
||
|
|
||
|
void PlcrFreeSharedProfiles(t_FmPcd *p_FmPcd, uint16_t numOfProfiles, uint16_t *profilesIds)
|
||
|
{
|
||
|
uint16_t i;
|
||
|
|
||
|
SANITY_CHECK_RETURN(p_FmPcd, E_INVALID_HANDLE);
|
||
|
|
||
|
ASSERT_COND(FmIsMaster(p_FmPcd->h_Fm));
|
||
|
for(i=0;i<numOfProfiles; i++)
|
||
|
{
|
||
|
ASSERT_COND(p_FmPcd->p_FmPcdPlcr->profiles[profilesIds[i]].profilesMng.allocated);
|
||
|
p_FmPcd->p_FmPcdPlcr->profiles[profilesIds[i]].profilesMng.allocated = FALSE;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void PlcrEnable(t_FmPcd *p_FmPcd)
|
||
|
{
|
||
|
t_FmPcdPlcrRegs *p_Regs = p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs;
|
||
|
|
||
|
WRITE_UINT32(p_Regs->fmpl_gcr, GET_UINT32(p_Regs->fmpl_gcr) | FM_PCD_PLCR_GCR_EN);
|
||
|
}
|
||
|
|
||
|
void PlcrDisable(t_FmPcd *p_FmPcd)
|
||
|
{
|
||
|
t_FmPcdPlcrRegs *p_Regs = p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs;
|
||
|
|
||
|
WRITE_UINT32(p_Regs->fmpl_gcr, GET_UINT32(p_Regs->fmpl_gcr) & ~FM_PCD_PLCR_GCR_EN);
|
||
|
}
|
||
|
|
||
|
t_Error FM_PCD_SetPlcrStatistics(t_Handle h_FmPcd, bool enable)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
uint32_t tmpReg32;
|
||
|
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd, E_INVALID_HANDLE);
|
||
|
SANITY_CHECK_RETURN_ERROR(!p_FmPcd->p_FmPcdDriverParam, E_INVALID_HANDLE);
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd->p_FmPcdPlcr, E_INVALID_HANDLE);
|
||
|
|
||
|
if(!FmIsMaster(p_FmPcd->h_Fm))
|
||
|
RETURN_ERROR(MAJOR, E_NOT_SUPPORTED, ("FM_PCD_SetPlcrStatistics - guest mode!"));
|
||
|
|
||
|
tmpReg32 = GET_UINT32(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs->fmpl_gcr);
|
||
|
if(enable)
|
||
|
tmpReg32 |= FM_PCD_PLCR_GCR_STEN;
|
||
|
else
|
||
|
tmpReg32 &= ~FM_PCD_PLCR_GCR_STEN;
|
||
|
|
||
|
WRITE_UINT32(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs->fmpl_gcr, tmpReg32);
|
||
|
return E_OK;
|
||
|
}
|
||
|
|
||
|
t_Error FM_PCD_ConfigPlcrAutoRefreshMode(t_Handle h_FmPcd, bool enable)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd, E_INVALID_HANDLE);
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd->p_FmPcdDriverParam, E_INVALID_HANDLE);
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd->p_FmPcdPlcr, E_INVALID_HANDLE);
|
||
|
|
||
|
if(!FmIsMaster(p_FmPcd->h_Fm))
|
||
|
RETURN_ERROR(MAJOR, E_NOT_SUPPORTED, ("FM_PCD_ConfigPlcrAutoRefreshMode - guest mode!"));
|
||
|
|
||
|
p_FmPcd->p_FmPcdDriverParam->plcrAutoRefresh = enable;
|
||
|
|
||
|
return E_OK;
|
||
|
}
|
||
|
|
||
|
|
||
|
t_Error FmPcdPlcrBuildProfile(t_Handle h_FmPcd, t_FmPcdPlcrProfileParams *p_Profile, t_FmPcdPlcrInterModuleProfileRegs *p_PlcrRegs)
|
||
|
{
|
||
|
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
t_Error err = E_OK;
|
||
|
uint32_t pemode, gnia, ynia, rnia;
|
||
|
|
||
|
/* Set G, Y, R Nia */
|
||
|
err = SetProfileNia(p_FmPcd, p_Profile->nextEngineOnGreen, &(p_Profile->paramsOnGreen), &gnia);
|
||
|
if(err)
|
||
|
RETURN_ERROR(MAJOR, err, NO_MSG);
|
||
|
err = SetProfileNia(p_FmPcd, p_Profile->nextEngineOnYellow, &(p_Profile->paramsOnYellow), &ynia);
|
||
|
if(err)
|
||
|
RETURN_ERROR(MAJOR, err, NO_MSG);
|
||
|
err = SetProfileNia(p_FmPcd, p_Profile->nextEngineOnRed, &(p_Profile->paramsOnRed), &rnia);
|
||
|
if(err)
|
||
|
RETURN_ERROR(MAJOR, err, NO_MSG);
|
||
|
|
||
|
|
||
|
/* Mode fmpl_pemode */
|
||
|
pemode = FM_PCD_PLCR_PEMODE_PI;
|
||
|
|
||
|
switch (p_Profile->algSelection)
|
||
|
{
|
||
|
case e_FM_PCD_PLCR_PASS_THROUGH:
|
||
|
p_PlcrRegs->fmpl_pecir = 0;
|
||
|
p_PlcrRegs->fmpl_pecbs = 0;
|
||
|
p_PlcrRegs->fmpl_pepepir_eir = 0;
|
||
|
p_PlcrRegs->fmpl_pepbs_ebs = 0;
|
||
|
p_PlcrRegs->fmpl_pelts = 0;
|
||
|
p_PlcrRegs->fmpl_pects = 0;
|
||
|
p_PlcrRegs->fmpl_pepts_ets = 0;
|
||
|
pemode &= ~FM_PCD_PLCR_PEMODE_ALG_MASK;
|
||
|
switch (p_Profile->colorMode)
|
||
|
{
|
||
|
case e_FM_PCD_PLCR_COLOR_BLIND:
|
||
|
pemode |= FM_PCD_PLCR_PEMODE_CBLND;
|
||
|
switch (p_Profile->color.dfltColor)
|
||
|
{
|
||
|
case e_FM_PCD_PLCR_GREEN:
|
||
|
pemode &= ~FM_PCD_PLCR_PEMODE_DEFC_MASK;
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_YELLOW:
|
||
|
pemode |= FM_PCD_PLCR_PEMODE_DEFC_Y;
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_RED:
|
||
|
pemode |= FM_PCD_PLCR_PEMODE_DEFC_R;
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_OVERRIDE:
|
||
|
pemode |= FM_PCD_PLCR_PEMODE_DEFC_OVERRIDE;
|
||
|
break;
|
||
|
default:
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_SELECTION, NO_MSG);
|
||
|
}
|
||
|
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_COLOR_AWARE:
|
||
|
pemode &= ~FM_PCD_PLCR_PEMODE_CBLND;
|
||
|
break;
|
||
|
default:
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_SELECTION, NO_MSG);
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case e_FM_PCD_PLCR_RFC_2698:
|
||
|
/* Select algorithm MODE[ALG] = "01" */
|
||
|
pemode |= FM_PCD_PLCR_PEMODE_ALG_RFC2698;
|
||
|
if (p_Profile->nonPassthroughAlgParams.comittedInfoRate > p_Profile->nonPassthroughAlgParams.peakOrAccessiveInfoRate)
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_SELECTION, ("in RFC2698 Peak rate must be equal or larger than comittedInfoRate."));
|
||
|
goto cont_rfc;
|
||
|
case e_FM_PCD_PLCR_RFC_4115:
|
||
|
/* Select algorithm MODE[ALG] = "10" */
|
||
|
pemode |= FM_PCD_PLCR_PEMODE_ALG_RFC4115;
|
||
|
cont_rfc:
|
||
|
/* Select Color-Blind / Color-Aware operation (MODE[CBLND]) */
|
||
|
switch (p_Profile->colorMode)
|
||
|
{
|
||
|
case e_FM_PCD_PLCR_COLOR_BLIND:
|
||
|
pemode |= FM_PCD_PLCR_PEMODE_CBLND;
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_COLOR_AWARE:
|
||
|
pemode &= ~FM_PCD_PLCR_PEMODE_CBLND;
|
||
|
/*In color aware more select override color interpretation (MODE[OVCLR]) */
|
||
|
switch (p_Profile->color.override)
|
||
|
{
|
||
|
case e_FM_PCD_PLCR_GREEN:
|
||
|
pemode &= ~FM_PCD_PLCR_PEMODE_OVCLR_MASK;
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_YELLOW:
|
||
|
pemode |= FM_PCD_PLCR_PEMODE_OVCLR_Y;
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_RED:
|
||
|
pemode |= FM_PCD_PLCR_PEMODE_OVCLR_R;
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_OVERRIDE:
|
||
|
pemode |= FM_PCD_PLCR_PEMODE_OVCLR_G_NC;
|
||
|
break;
|
||
|
default:
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_SELECTION, NO_MSG);
|
||
|
}
|
||
|
break;
|
||
|
default:
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_SELECTION, NO_MSG);
|
||
|
}
|
||
|
/* Select Measurement Unit Mode to BYTE or PACKET (MODE[PKT]) */
|
||
|
switch (p_Profile->nonPassthroughAlgParams.rateMode)
|
||
|
{
|
||
|
case e_FM_PCD_PLCR_BYTE_MODE :
|
||
|
pemode &= ~FM_PCD_PLCR_PEMODE_PKT;
|
||
|
switch (p_Profile->nonPassthroughAlgParams.byteModeParams.frameLengthSelection)
|
||
|
{
|
||
|
case e_FM_PCD_PLCR_L2_FRM_LEN:
|
||
|
pemode |= FM_PCD_PLCR_PEMODE_FLS_L2;
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_L3_FRM_LEN:
|
||
|
pemode |= FM_PCD_PLCR_PEMODE_FLS_L3;
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_L4_FRM_LEN:
|
||
|
pemode |= FM_PCD_PLCR_PEMODE_FLS_L4;
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_FULL_FRM_LEN:
|
||
|
pemode |= FM_PCD_PLCR_PEMODE_FLS_FULL;
|
||
|
break;
|
||
|
default:
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_SELECTION, NO_MSG);
|
||
|
}
|
||
|
switch (p_Profile->nonPassthroughAlgParams.byteModeParams.rollBackFrameSelection)
|
||
|
{
|
||
|
case e_FM_PCD_PLCR_ROLLBACK_L2_FRM_LEN:
|
||
|
pemode &= ~FM_PCD_PLCR_PEMODE_RBFLS;
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_ROLLBACK_FULL_FRM_LEN:
|
||
|
pemode |= FM_PCD_PLCR_PEMODE_RBFLS;
|
||
|
break;
|
||
|
default:
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_SELECTION, NO_MSG);
|
||
|
}
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_PACKET_MODE :
|
||
|
pemode |= FM_PCD_PLCR_PEMODE_PKT;
|
||
|
break;
|
||
|
default:
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_SELECTION, NO_MSG);
|
||
|
}
|
||
|
/* Select timeStamp floating point position (MODE[FPP]) to fit the actual traffic rates. For PACKET
|
||
|
mode with low traffic rates move the fixed point to the left to increase fraction accuracy. For BYTE
|
||
|
mode with high traffic rates move the fixed point to the right to increase integer accuracy. */
|
||
|
|
||
|
/* Configure Traffic Parameters*/
|
||
|
{
|
||
|
uint32_t cir=0, cbs=0, pir_eir=0, pbs_ebs=0, fpp=0;
|
||
|
|
||
|
calcRates(h_FmPcd, &p_Profile->nonPassthroughAlgParams, &cir, &cbs, &pir_eir, &pbs_ebs, &fpp);
|
||
|
|
||
|
/* Set Committed Information Rate (CIR) */
|
||
|
p_PlcrRegs->fmpl_pecir = cir;
|
||
|
/* Set Committed Burst Size (CBS). */
|
||
|
p_PlcrRegs->fmpl_pecbs = cbs;
|
||
|
/* Set Peak Information Rate (PIR_EIR used as PIR) */
|
||
|
p_PlcrRegs->fmpl_pepepir_eir = pir_eir;
|
||
|
/* Set Peak Burst Size (PBS_EBS used as PBS) */
|
||
|
p_PlcrRegs->fmpl_pepbs_ebs = pbs_ebs;
|
||
|
|
||
|
/* Initialize the Metering Buckets to be full (write them with 0xFFFFFFFF. */
|
||
|
/* Peak Rate Token Bucket Size (PTS_ETS used as PTS) */
|
||
|
p_PlcrRegs->fmpl_pepts_ets = 0xFFFFFFFF;
|
||
|
/* Committed Rate Token Bucket Size (CTS) */
|
||
|
p_PlcrRegs->fmpl_pects = 0xFFFFFFFF;
|
||
|
|
||
|
/* Set the FPP based on calculation */
|
||
|
pemode |= (fpp << FM_PCD_PLCR_PEMODE_FPP_SHIFT);
|
||
|
}
|
||
|
break; /* FM_PCD_PLCR_PEMODE_ALG_RFC2698 , FM_PCD_PLCR_PEMODE_ALG_RFC4115 */
|
||
|
default:
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_SELECTION, NO_MSG);
|
||
|
}
|
||
|
|
||
|
p_PlcrRegs->fmpl_pemode = pemode;
|
||
|
|
||
|
p_PlcrRegs->fmpl_pegnia = gnia;
|
||
|
p_PlcrRegs->fmpl_peynia = ynia;
|
||
|
p_PlcrRegs->fmpl_pernia = rnia;
|
||
|
|
||
|
/* Zero Counters */
|
||
|
p_PlcrRegs->fmpl_pegpc = 0;
|
||
|
p_PlcrRegs->fmpl_peypc = 0;
|
||
|
p_PlcrRegs->fmpl_perpc = 0;
|
||
|
p_PlcrRegs->fmpl_perypc = 0;
|
||
|
p_PlcrRegs->fmpl_perrpc = 0;
|
||
|
|
||
|
return E_OK;
|
||
|
}
|
||
|
|
||
|
void FmPcdPlcrValidateProfileSw(t_Handle h_FmPcd, uint16_t absoluteProfileId)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
|
||
|
ASSERT_COND(!p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].valid);
|
||
|
p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].valid = TRUE;
|
||
|
}
|
||
|
|
||
|
void FmPcdPlcrInvalidateProfileSw(t_Handle h_FmPcd, uint16_t absoluteProfileId)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
|
||
|
ASSERT_COND(p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].valid);
|
||
|
p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].valid = FALSE;
|
||
|
}
|
||
|
|
||
|
t_Handle PlcrConfig(t_FmPcd *p_FmPcd, t_FmPcdParams *p_FmPcdParams)
|
||
|
{
|
||
|
t_FmPcdPlcr *p_FmPcdPlcr;
|
||
|
/*uint8_t i=0;*/
|
||
|
|
||
|
UNUSED(p_FmPcd);
|
||
|
UNUSED(p_FmPcdParams);
|
||
|
|
||
|
p_FmPcdPlcr = (t_FmPcdPlcr *) XX_Malloc(sizeof(t_FmPcdPlcr));
|
||
|
if (!p_FmPcdPlcr)
|
||
|
{
|
||
|
REPORT_ERROR(MAJOR, E_NO_MEMORY, ("FM Policer structure allocation FAILED"));
|
||
|
return NULL;
|
||
|
}
|
||
|
memset(p_FmPcdPlcr, 0, sizeof(t_FmPcdPlcr));
|
||
|
if(p_FmPcd->guestId == NCSW_MASTER_ID)
|
||
|
{
|
||
|
p_FmPcdPlcr->p_FmPcdPlcrRegs = (t_FmPcdPlcrRegs *)UINT_TO_PTR(FmGetPcdPlcrBaseAddr(p_FmPcdParams->h_Fm));
|
||
|
p_FmPcd->p_FmPcdDriverParam->plcrAutoRefresh = DEFAULT_plcrAutoRefresh;
|
||
|
p_FmPcd->exceptions |= (DEFAULT_fmPcdPlcrExceptions | DEFAULT_fmPcdPlcrErrorExceptions);
|
||
|
}
|
||
|
|
||
|
p_FmPcdPlcr->numOfSharedProfiles = DEFAULT_numOfSharedPlcrProfiles;
|
||
|
|
||
|
return p_FmPcdPlcr;
|
||
|
}
|
||
|
|
||
|
t_Error PlcrInit(t_FmPcd *p_FmPcd)
|
||
|
{
|
||
|
t_FmPcdDriverParam *p_Param = p_FmPcd->p_FmPcdDriverParam;
|
||
|
t_FmPcdPlcr *p_FmPcdPlcr = p_FmPcd->p_FmPcdPlcr;
|
||
|
uint32_t tmpReg32 = 0;
|
||
|
t_Error err = E_OK;
|
||
|
t_FmPcdPlcrRegs *p_Regs = p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs;
|
||
|
t_FmPcdIpcMsg msg;
|
||
|
uint32_t replyLength;
|
||
|
t_FmPcdIpcReply reply;
|
||
|
|
||
|
if ((p_FmPcd->guestId != NCSW_MASTER_ID) &&
|
||
|
(p_FmPcdPlcr->numOfSharedProfiles))
|
||
|
{
|
||
|
int i, j, index = 0;
|
||
|
uint32_t walking1Mask = 0x80000000;
|
||
|
uint32_t sharedProfilesMask[FM_PCD_PLCR_NUM_ENTRIES/32];
|
||
|
|
||
|
memset(sharedProfilesMask, 0, FM_PCD_PLCR_NUM_ENTRIES/32 * sizeof(uint32_t));
|
||
|
memset(&reply, 0, sizeof(reply));
|
||
|
memset(&msg, 0, sizeof(msg));
|
||
|
msg.msgId = FM_PCD_ALLOC_SHARED_PROFILES;
|
||
|
memcpy(msg.msgBody, (uint8_t *)&p_FmPcdPlcr->numOfSharedProfiles, sizeof(uint16_t));
|
||
|
replyLength = sizeof(uint32_t) + sizeof(sharedProfilesMask);
|
||
|
if ((err = XX_IpcSendMessage(p_FmPcd->h_IpcSession,
|
||
|
(uint8_t*)&msg,
|
||
|
sizeof(msg.msgId)+ sizeof(p_FmPcdPlcr->numOfSharedProfiles),
|
||
|
(uint8_t*)&reply,
|
||
|
&replyLength,
|
||
|
NULL,
|
||
|
NULL)) != E_OK)
|
||
|
RETURN_ERROR(MAJOR, err,NO_MSG);
|
||
|
if (replyLength != (sizeof(uint32_t) + sizeof(sharedProfilesMask)))
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_VALUE, ("IPC reply length mismatch"));
|
||
|
|
||
|
memcpy(sharedProfilesMask, reply.replyBody, sizeof(sharedProfilesMask));
|
||
|
/* translate 8 regs of 32 bits masks into an array of up to 256 indexes. */
|
||
|
for(i = 0; i<FM_PCD_PLCR_NUM_ENTRIES/32 ; i++)
|
||
|
{
|
||
|
if(sharedProfilesMask[i])
|
||
|
{
|
||
|
for(j = 0 ; j<32 ; j++)
|
||
|
{
|
||
|
if(sharedProfilesMask[i] & walking1Mask)
|
||
|
p_FmPcd->p_FmPcdPlcr->sharedProfilesIds[index++] = (uint16_t)(i*32+j);
|
||
|
walking1Mask >>= 1;
|
||
|
}
|
||
|
walking1Mask = 0x80000000;
|
||
|
}
|
||
|
}
|
||
|
return (t_Error)reply.error;
|
||
|
}
|
||
|
|
||
|
if(p_FmPcdPlcr->numOfSharedProfiles)
|
||
|
{
|
||
|
err = PlcrAllocSharedProfiles(p_FmPcd, p_FmPcdPlcr->numOfSharedProfiles, p_FmPcd->p_FmPcdPlcr->sharedProfilesIds);
|
||
|
if(err)
|
||
|
RETURN_ERROR(MAJOR, err,NO_MSG);
|
||
|
}
|
||
|
|
||
|
/**********************FMPL_GCR******************/
|
||
|
tmpReg32 = 0;
|
||
|
tmpReg32 |= FM_PCD_PLCR_GCR_STEN;
|
||
|
if(p_Param->plcrAutoRefresh)
|
||
|
tmpReg32 |= FM_PCD_PLCR_GCR_DAR;
|
||
|
tmpReg32 |= NIA_ENG_BMI | NIA_BMI_AC_ENQ_FRAME;
|
||
|
|
||
|
WRITE_UINT32(p_Regs->fmpl_gcr, tmpReg32);
|
||
|
/**********************FMPL_GCR******************/
|
||
|
|
||
|
/**********************FMPL_EEVR******************/
|
||
|
WRITE_UINT32(p_Regs->fmpl_eevr, (FM_PCD_PLCR_DOUBLE_ECC | FM_PCD_PLCR_INIT_ENTRY_ERROR));
|
||
|
/**********************FMPL_EEVR******************/
|
||
|
/**********************FMPL_EIER******************/
|
||
|
tmpReg32 = 0;
|
||
|
if(p_FmPcd->exceptions & FM_PCD_EX_PLCR_DOUBLE_ECC)
|
||
|
{
|
||
|
FmEnableRamsEcc(p_FmPcd->h_Fm);
|
||
|
tmpReg32 |= FM_PCD_PLCR_DOUBLE_ECC;
|
||
|
}
|
||
|
if(p_FmPcd->exceptions & FM_PCD_EX_PLCR_INIT_ENTRY_ERROR)
|
||
|
tmpReg32 |= FM_PCD_PLCR_INIT_ENTRY_ERROR;
|
||
|
WRITE_UINT32(p_Regs->fmpl_eier, tmpReg32);
|
||
|
/**********************FMPL_EIER******************/
|
||
|
|
||
|
/**********************FMPL_EVR******************/
|
||
|
WRITE_UINT32(p_Regs->fmpl_evr, (FM_PCD_PLCR_PRAM_SELF_INIT_COMPLETE | FM_PCD_PLCR_ATOMIC_ACTION_COMPLETE));
|
||
|
/**********************FMPL_EVR******************/
|
||
|
/**********************FMPL_IER******************/
|
||
|
tmpReg32 = 0;
|
||
|
if(p_FmPcd->exceptions & FM_PCD_EX_PLCR_PRAM_SELF_INIT_COMPLETE)
|
||
|
tmpReg32 |= FM_PCD_PLCR_PRAM_SELF_INIT_COMPLETE;
|
||
|
if(p_FmPcd->exceptions & FM_PCD_EX_PLCR_ATOMIC_ACTION_COMPLETE )
|
||
|
tmpReg32 |= FM_PCD_PLCR_ATOMIC_ACTION_COMPLETE;
|
||
|
WRITE_UINT32(p_Regs->fmpl_ier, tmpReg32);
|
||
|
/**********************FMPL_IER******************/
|
||
|
|
||
|
/* register even if no interrupts enabled, to allow future enablement */
|
||
|
FmRegisterIntr(p_FmPcd->h_Fm, e_FM_MOD_PLCR, 0, e_FM_INTR_TYPE_ERR, PcdPlcrErrorException, p_FmPcd);
|
||
|
FmRegisterIntr(p_FmPcd->h_Fm, e_FM_MOD_PLCR, 0, e_FM_INTR_TYPE_NORMAL, PcdPlcrException, p_FmPcd);
|
||
|
|
||
|
/* driver initializes one DFLT profile at the last entry*/
|
||
|
/**********************FMPL_DPMR******************/
|
||
|
tmpReg32 = 0;
|
||
|
WRITE_UINT32(p_Regs->fmpl_dpmr, tmpReg32);
|
||
|
p_FmPcd->p_FmPcdPlcr->profiles[0].profilesMng.allocated = TRUE;
|
||
|
|
||
|
return E_OK;
|
||
|
}
|
||
|
|
||
|
t_Error PlcrFree(t_FmPcd *p_FmPcd)
|
||
|
{
|
||
|
t_Error err;
|
||
|
t_FmPcdIpcSharedPlcrAllocParams ipcSharedPlcrParams;
|
||
|
t_FmPcdIpcMsg msg;
|
||
|
|
||
|
FmUnregisterIntr(p_FmPcd->h_Fm, e_FM_MOD_PLCR, 0, e_FM_INTR_TYPE_ERR);
|
||
|
FmUnregisterIntr(p_FmPcd->h_Fm, e_FM_MOD_PLCR, 0, e_FM_INTR_TYPE_NORMAL);
|
||
|
|
||
|
if(p_FmPcd->p_FmPcdPlcr->numOfSharedProfiles)
|
||
|
{
|
||
|
if(p_FmPcd->guestId != NCSW_MASTER_ID)
|
||
|
{
|
||
|
int i;
|
||
|
memset(ipcSharedPlcrParams.sharedProfilesMask, 0, sizeof(ipcSharedPlcrParams.sharedProfilesMask));
|
||
|
/* Free resources using IPC messaging */
|
||
|
ipcSharedPlcrParams.num = p_FmPcd->p_FmPcdPlcr->numOfSharedProfiles;
|
||
|
|
||
|
/* translate the allocated profile id's to a 32bit * 8regs mask */
|
||
|
for(i = 0;i<p_FmPcd->p_FmPcdPlcr->numOfSharedProfiles;i++)
|
||
|
ipcSharedPlcrParams.sharedProfilesMask[p_FmPcd->p_FmPcdPlcr->sharedProfilesIds[i]/32] |= (0x80000000 >> (p_FmPcd->p_FmPcdPlcr->sharedProfilesIds[i] % 32));
|
||
|
|
||
|
memset(&msg, 0, sizeof(msg));
|
||
|
msg.msgId = FM_PCD_FREE_SHARED_PROFILES;
|
||
|
memcpy(msg.msgBody, &ipcSharedPlcrParams, sizeof(ipcSharedPlcrParams));
|
||
|
if ((err = XX_IpcSendMessage(p_FmPcd->h_IpcSession,
|
||
|
(uint8_t*)&msg,
|
||
|
sizeof(msg.msgId)+sizeof(ipcSharedPlcrParams),
|
||
|
NULL,
|
||
|
NULL,
|
||
|
NULL,
|
||
|
NULL)) != E_OK)
|
||
|
RETURN_ERROR(MAJOR, err,NO_MSG);
|
||
|
}
|
||
|
/* else
|
||
|
PlcrFreeSharedProfiles(p_FmPcd, p_FmPcd->p_FmPcdPlcr->numOfSharedProfiles, p_FmPcd->p_FmPcdPlcr->sharedProfilesIds);*/
|
||
|
}
|
||
|
return E_OK;
|
||
|
}
|
||
|
|
||
|
t_Error FmPcdPlcrGetAbsoluteProfileId(t_Handle h_FmPcd,
|
||
|
e_FmPcdProfileTypeSelection profileType,
|
||
|
t_Handle h_FmPort,
|
||
|
uint16_t relativeProfile,
|
||
|
uint16_t *p_AbsoluteId)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
t_FmPcdPlcr *p_FmPcdPlcr = p_FmPcd->p_FmPcdPlcr;
|
||
|
uint8_t i;
|
||
|
|
||
|
switch (profileType)
|
||
|
{
|
||
|
case e_FM_PCD_PLCR_PORT_PRIVATE:
|
||
|
/* get port PCD id from port handle */
|
||
|
for(i=0;i<FM_MAX_NUM_OF_PORTS;i++)
|
||
|
if(p_FmPcd->p_FmPcdPlcr->portsMapping[i].h_FmPort == h_FmPort)
|
||
|
break;
|
||
|
if (i == FM_MAX_NUM_OF_PORTS)
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_STATE , ("Invalid port handle."));
|
||
|
|
||
|
if(!p_FmPcd->p_FmPcdPlcr->portsMapping[i].numOfProfiles)
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_SELECTION , ("Port has no allocated profiles"));
|
||
|
if(relativeProfile >= p_FmPcd->p_FmPcdPlcr->portsMapping[i].numOfProfiles)
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_SELECTION , ("Profile id is out of range"));
|
||
|
*p_AbsoluteId = (uint16_t)(p_FmPcd->p_FmPcdPlcr->portsMapping[i].profilesBase + relativeProfile);
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_SHARED:
|
||
|
if(relativeProfile >= p_FmPcdPlcr->numOfSharedProfiles)
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_SELECTION , ("Profile id is out of range"));
|
||
|
*p_AbsoluteId = (uint16_t)(p_FmPcdPlcr->sharedProfilesIds[relativeProfile]);
|
||
|
break;
|
||
|
default:
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_SELECTION, ("Invalid policer profile type"));
|
||
|
}
|
||
|
return E_OK;
|
||
|
}
|
||
|
|
||
|
uint16_t FmPcdPlcrGetPortProfilesBase(t_Handle h_FmPcd, uint8_t hardwarePortId)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd *)h_FmPcd;
|
||
|
uint16_t swPortIndex = 0;
|
||
|
|
||
|
HW_PORT_ID_TO_SW_PORT_INDX(swPortIndex, hardwarePortId);
|
||
|
|
||
|
return p_FmPcd->p_FmPcdPlcr->portsMapping[swPortIndex].profilesBase;
|
||
|
}
|
||
|
|
||
|
uint16_t FmPcdPlcrGetPortNumOfProfiles(t_Handle h_FmPcd, uint8_t hardwarePortId)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd *)h_FmPcd;
|
||
|
uint16_t swPortIndex = 0;
|
||
|
|
||
|
HW_PORT_ID_TO_SW_PORT_INDX(swPortIndex, hardwarePortId);
|
||
|
|
||
|
return p_FmPcd->p_FmPcdPlcr->portsMapping[swPortIndex].numOfProfiles;
|
||
|
|
||
|
}
|
||
|
uint32_t FmPcdPlcrBuildWritePlcrActionReg(uint16_t absoluteProfileId)
|
||
|
{
|
||
|
return (uint32_t)(FM_PCD_PLCR_PAR_GO |
|
||
|
((uint32_t)absoluteProfileId << FM_PCD_PLCR_PAR_PNUM_SHIFT));
|
||
|
}
|
||
|
|
||
|
uint32_t FmPcdPlcrBuildWritePlcrActionRegs(uint16_t absoluteProfileId)
|
||
|
{
|
||
|
return (uint32_t)(FM_PCD_PLCR_PAR_GO |
|
||
|
((uint32_t)absoluteProfileId << FM_PCD_PLCR_PAR_PNUM_SHIFT) |
|
||
|
FM_PCD_PLCR_PAR_PWSEL_MASK);
|
||
|
}
|
||
|
|
||
|
bool FmPcdPlcrHwProfileIsValid(uint32_t profileModeReg)
|
||
|
{
|
||
|
|
||
|
if(profileModeReg & FM_PCD_PLCR_PEMODE_PI)
|
||
|
return TRUE;
|
||
|
else
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
uint32_t FmPcdPlcrBuildReadPlcrActionReg(uint16_t absoluteProfileId)
|
||
|
{
|
||
|
return (uint32_t)(FM_PCD_PLCR_PAR_GO |
|
||
|
FM_PCD_PLCR_PAR_R |
|
||
|
((uint32_t)absoluteProfileId << FM_PCD_PLCR_PAR_PNUM_SHIFT) |
|
||
|
FM_PCD_PLCR_PAR_PWSEL_MASK);
|
||
|
}
|
||
|
|
||
|
uint32_t FmPcdPlcrBuildCounterProfileReg(e_FmPcdPlcrProfileCounters counter)
|
||
|
{
|
||
|
switch(counter)
|
||
|
{
|
||
|
case(e_FM_PCD_PLCR_PROFILE_GREEN_PACKET_TOTAL_COUNTER):
|
||
|
return FM_PCD_PLCR_PAR_PWSEL_PEGPC;
|
||
|
case(e_FM_PCD_PLCR_PROFILE_YELLOW_PACKET_TOTAL_COUNTER):
|
||
|
return FM_PCD_PLCR_PAR_PWSEL_PEYPC;
|
||
|
case(e_FM_PCD_PLCR_PROFILE_RED_PACKET_TOTAL_COUNTER) :
|
||
|
return FM_PCD_PLCR_PAR_PWSEL_PERPC;
|
||
|
case(e_FM_PCD_PLCR_PROFILE_RECOLOURED_YELLOW_PACKET_TOTAL_COUNTER) :
|
||
|
return FM_PCD_PLCR_PAR_PWSEL_PERYPC;
|
||
|
case(e_FM_PCD_PLCR_PROFILE_RECOLOURED_RED_PACKET_TOTAL_COUNTER) :
|
||
|
return FM_PCD_PLCR_PAR_PWSEL_PERRPC;
|
||
|
default:
|
||
|
REPORT_ERROR(MAJOR, E_INVALID_SELECTION, NO_MSG);
|
||
|
return 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
uint32_t FmPcdPlcrBuildNiaProfileReg(bool green, bool yellow, bool red)
|
||
|
{
|
||
|
|
||
|
uint32_t tmpReg32 = 0;
|
||
|
|
||
|
if(green)
|
||
|
tmpReg32 |= FM_PCD_PLCR_PAR_PWSEL_PEGNIA;
|
||
|
if(yellow)
|
||
|
tmpReg32 |= FM_PCD_PLCR_PAR_PWSEL_PEYNIA;
|
||
|
if(red)
|
||
|
tmpReg32 |= FM_PCD_PLCR_PAR_PWSEL_PERNIA;
|
||
|
|
||
|
return tmpReg32;
|
||
|
}
|
||
|
|
||
|
void FmPcdPlcrUpdateRequiredAction(t_Handle h_FmPcd, uint16_t absoluteProfileId, uint32_t requiredAction)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
|
||
|
ASSERT_COND(p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].valid);
|
||
|
|
||
|
p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].requiredAction = requiredAction;
|
||
|
}
|
||
|
|
||
|
t_Error FmPcdPlcrProfileTryLock(t_Handle h_FmPcd, uint16_t profileId, bool intr)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd *)h_FmPcd;
|
||
|
bool ans;
|
||
|
if (intr)
|
||
|
ans = TRY_LOCK(NULL, &p_FmPcd->p_FmPcdPlcr->profiles[profileId].lock);
|
||
|
else
|
||
|
ans = TRY_LOCK(p_FmPcd->h_Spinlock, &p_FmPcd->p_FmPcdPlcr->profiles[profileId].lock);
|
||
|
if (ans)
|
||
|
return E_OK;
|
||
|
return ERROR_CODE(E_BUSY);
|
||
|
}
|
||
|
|
||
|
void FmPcdPlcrReleaseProfileLock(t_Handle h_FmPcd, uint16_t profileId)
|
||
|
{
|
||
|
RELEASE_LOCK(((t_FmPcd*)h_FmPcd)->p_FmPcdPlcr->profiles[profileId].lock);
|
||
|
}
|
||
|
|
||
|
/**************************************************/
|
||
|
/*............Policer API.........................*/
|
||
|
/**************************************************/
|
||
|
|
||
|
t_Handle FM_PCD_PlcrSetProfile(t_Handle h_FmPcd,
|
||
|
t_FmPcdPlcrProfileParams *p_Profile)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
t_FmPcdPlcrRegs *p_FmPcdPlcrRegs;
|
||
|
t_FmPcdPlcrInterModuleProfileRegs plcrProfileReg;
|
||
|
uint32_t intFlags;
|
||
|
uint16_t absoluteProfileId;
|
||
|
t_Error err = E_OK;
|
||
|
uint32_t tmpReg32;
|
||
|
|
||
|
SANITY_CHECK_RETURN_VALUE(p_FmPcd, E_INVALID_HANDLE, NULL);
|
||
|
SANITY_CHECK_RETURN_VALUE(!p_FmPcd->p_FmPcdDriverParam, E_INVALID_STATE, NULL);
|
||
|
SANITY_CHECK_RETURN_VALUE(p_FmPcd->p_FmPcdPlcr, E_INVALID_HANDLE, NULL);
|
||
|
|
||
|
if (p_FmPcd->h_Hc)
|
||
|
return FmHcPcdPlcrSetProfile(p_FmPcd->h_Hc, p_Profile);
|
||
|
|
||
|
p_FmPcdPlcrRegs = p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs;
|
||
|
SANITY_CHECK_RETURN_VALUE(p_FmPcdPlcrRegs, E_INVALID_HANDLE, NULL);
|
||
|
|
||
|
if (p_Profile->modify)
|
||
|
{
|
||
|
absoluteProfileId = (uint16_t)(PTR_TO_UINT(p_Profile->id.h_Profile)-1);
|
||
|
if (absoluteProfileId >= FM_PCD_PLCR_NUM_ENTRIES)
|
||
|
{
|
||
|
REPORT_ERROR(MAJOR, E_INVALID_VALUE, ("profileId too Big "));
|
||
|
return NULL;
|
||
|
}
|
||
|
if (FmPcdPlcrProfileTryLock(p_FmPcd, absoluteProfileId, FALSE))
|
||
|
return NULL;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
intFlags = FmPcdLock(p_FmPcd);
|
||
|
err = FmPcdPlcrGetAbsoluteProfileId(h_FmPcd,
|
||
|
p_Profile->id.newParams.profileType,
|
||
|
p_Profile->id.newParams.h_FmPort,
|
||
|
p_Profile->id.newParams.relativeProfileId,
|
||
|
&absoluteProfileId);
|
||
|
if (absoluteProfileId >= FM_PCD_PLCR_NUM_ENTRIES)
|
||
|
{
|
||
|
REPORT_ERROR(MAJOR, E_INVALID_VALUE, ("profileId too Big "));
|
||
|
return NULL;
|
||
|
}
|
||
|
if(err)
|
||
|
{
|
||
|
FmPcdUnlock(p_FmPcd, intFlags);
|
||
|
REPORT_ERROR(MAJOR, err, NO_MSG);
|
||
|
return NULL;
|
||
|
}
|
||
|
err = FmPcdPlcrProfileTryLock(p_FmPcd, absoluteProfileId, TRUE);
|
||
|
FmPcdUnlock(p_FmPcd, intFlags);
|
||
|
if (err)
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
/* if no override, check first that this profile is unused */
|
||
|
if(!p_Profile->modify)
|
||
|
{
|
||
|
/* read specified profile into profile registers */
|
||
|
tmpReg32 = FmPcdPlcrBuildReadPlcrActionReg(absoluteProfileId);
|
||
|
intFlags = FmPcdLock(p_FmPcd);
|
||
|
WritePar(p_FmPcd, tmpReg32);
|
||
|
tmpReg32 = GET_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pemode);
|
||
|
FmPcdUnlock(p_FmPcd, intFlags);
|
||
|
if (tmpReg32 & FM_PCD_PLCR_PEMODE_PI)
|
||
|
{
|
||
|
RELEASE_LOCK(p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].lock);
|
||
|
REPORT_ERROR(MAJOR, E_ALREADY_EXISTS, ("Policer Profile is already used"));
|
||
|
return NULL;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
memset(&plcrProfileReg, 0, sizeof(t_FmPcdPlcrInterModuleProfileRegs));
|
||
|
|
||
|
err = FmPcdPlcrBuildProfile(h_FmPcd, p_Profile, &plcrProfileReg);
|
||
|
if(err)
|
||
|
{
|
||
|
RELEASE_LOCK(p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].lock);
|
||
|
REPORT_ERROR(MAJOR, err, NO_MSG);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].nextEngineOnGreen = p_Profile->nextEngineOnGreen;
|
||
|
memcpy(&p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].paramsOnGreen, &(p_Profile->paramsOnGreen), sizeof(u_FmPcdPlcrNextEngineParams));
|
||
|
|
||
|
p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].nextEngineOnYellow = p_Profile->nextEngineOnYellow;
|
||
|
memcpy(&p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].paramsOnYellow, &(p_Profile->paramsOnYellow), sizeof(u_FmPcdPlcrNextEngineParams));
|
||
|
|
||
|
p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].nextEngineOnRed = p_Profile->nextEngineOnRed;
|
||
|
memcpy(&p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].paramsOnRed, &(p_Profile->paramsOnRed), sizeof(u_FmPcdPlcrNextEngineParams));
|
||
|
|
||
|
intFlags = FmPcdLock(p_FmPcd);
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pemode , plcrProfileReg.fmpl_pemode);
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pegnia , plcrProfileReg.fmpl_pegnia);
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_peynia , plcrProfileReg.fmpl_peynia);
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pernia , plcrProfileReg.fmpl_pernia);
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pecir , plcrProfileReg.fmpl_pecir);
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pecbs , plcrProfileReg.fmpl_pecbs);
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pepepir_eir,plcrProfileReg.fmpl_pepepir_eir);
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pepbs_ebs,plcrProfileReg.fmpl_pepbs_ebs);
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pelts , plcrProfileReg.fmpl_pelts);
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pects , plcrProfileReg.fmpl_pects);
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pepts_ets,plcrProfileReg.fmpl_pepts_ets);
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pegpc , plcrProfileReg.fmpl_pegpc);
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_peypc , plcrProfileReg.fmpl_peypc);
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_perpc , plcrProfileReg.fmpl_perpc);
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_perypc , plcrProfileReg.fmpl_perypc);
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_perrpc , plcrProfileReg.fmpl_perrpc);
|
||
|
|
||
|
tmpReg32 = FmPcdPlcrBuildWritePlcrActionRegs(absoluteProfileId);
|
||
|
WritePar(p_FmPcd, tmpReg32);
|
||
|
|
||
|
FmPcdUnlock(p_FmPcd, intFlags);
|
||
|
|
||
|
if (!p_Profile->modify)
|
||
|
FmPcdPlcrValidateProfileSw(p_FmPcd,absoluteProfileId);
|
||
|
|
||
|
RELEASE_LOCK(p_FmPcd->p_FmPcdPlcr->profiles[absoluteProfileId].lock);
|
||
|
|
||
|
return UINT_TO_PTR((uint64_t)absoluteProfileId+1);
|
||
|
}
|
||
|
|
||
|
t_Error FM_PCD_PlcrDeleteProfile(t_Handle h_FmPcd, t_Handle h_Profile)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
uint16_t profileIndx = (uint16_t)(PTR_TO_UINT(h_Profile)-1);
|
||
|
uint32_t tmpReg32, intFlags;
|
||
|
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd, E_INVALID_HANDLE);
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd->p_FmPcdPlcr, E_INVALID_HANDLE);
|
||
|
SANITY_CHECK_RETURN_ERROR((profileIndx < FM_PCD_PLCR_NUM_ENTRIES), E_INVALID_SELECTION);
|
||
|
|
||
|
if (p_FmPcd->h_Hc)
|
||
|
return FmHcPcdPlcrDeleteProfile(p_FmPcd->h_Hc, h_Profile);
|
||
|
|
||
|
FmPcdPlcrInvalidateProfileSw(p_FmPcd,profileIndx);
|
||
|
|
||
|
intFlags = FmPcdLock(p_FmPcd);
|
||
|
WRITE_UINT32(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs->profileRegs.fmpl_pemode, ~FM_PCD_PLCR_PEMODE_PI);
|
||
|
|
||
|
tmpReg32 = FmPcdPlcrBuildWritePlcrActionRegs(profileIndx);
|
||
|
WritePar(p_FmPcd, tmpReg32);
|
||
|
FmPcdUnlock(p_FmPcd, intFlags);
|
||
|
|
||
|
return E_OK;
|
||
|
}
|
||
|
|
||
|
/* ......... */
|
||
|
/***************************************************/
|
||
|
/*............Policer Profile Counter..............*/
|
||
|
/***************************************************/
|
||
|
uint32_t FM_PCD_PlcrGetProfileCounter(t_Handle h_FmPcd, t_Handle h_Profile, e_FmPcdPlcrProfileCounters counter)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
uint16_t profileIndx = (uint16_t)(PTR_TO_UINT(h_Profile)-1);
|
||
|
t_FmPcdPlcrRegs *p_FmPcdPlcrRegs;
|
||
|
uint32_t intFlags, counterVal = 0;
|
||
|
|
||
|
SANITY_CHECK_RETURN_VALUE(p_FmPcd, E_INVALID_HANDLE, 0);
|
||
|
SANITY_CHECK_RETURN_VALUE(!p_FmPcd->p_FmPcdDriverParam, E_INVALID_STATE, 0);
|
||
|
SANITY_CHECK_RETURN_VALUE(p_FmPcd->p_FmPcdPlcr, E_INVALID_HANDLE, 0);
|
||
|
|
||
|
if (p_FmPcd->h_Hc)
|
||
|
return FmHcPcdPlcrGetProfileCounter(p_FmPcd->h_Hc, h_Profile, counter);
|
||
|
|
||
|
p_FmPcdPlcrRegs = p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs;
|
||
|
SANITY_CHECK_RETURN_VALUE(p_FmPcdPlcrRegs, E_INVALID_HANDLE, 0);
|
||
|
|
||
|
if (profileIndx >= FM_PCD_PLCR_NUM_ENTRIES)
|
||
|
{
|
||
|
REPORT_ERROR(MAJOR, E_INVALID_VALUE, ("profileId too Big "));
|
||
|
return 0;
|
||
|
}
|
||
|
intFlags = FmPcdLock(p_FmPcd);
|
||
|
WritePar(p_FmPcd, FmPcdPlcrBuildReadPlcrActionReg(profileIndx));
|
||
|
|
||
|
if(!FmPcdPlcrHwProfileIsValid(GET_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pemode)))
|
||
|
{
|
||
|
REPORT_ERROR(MAJOR, E_INVALID_VALUE, ("Uninitialized profile"));
|
||
|
FmPcdUnlock(p_FmPcd, intFlags);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
switch (counter)
|
||
|
{
|
||
|
case e_FM_PCD_PLCR_PROFILE_GREEN_PACKET_TOTAL_COUNTER:
|
||
|
counterVal = (GET_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pegpc));
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_PROFILE_YELLOW_PACKET_TOTAL_COUNTER:
|
||
|
counterVal = GET_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_peypc);
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_PROFILE_RED_PACKET_TOTAL_COUNTER:
|
||
|
counterVal = GET_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_perpc);
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_PROFILE_RECOLOURED_YELLOW_PACKET_TOTAL_COUNTER:
|
||
|
counterVal = GET_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_perypc);
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_PROFILE_RECOLOURED_RED_PACKET_TOTAL_COUNTER:
|
||
|
counterVal = GET_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_perrpc);
|
||
|
break;
|
||
|
default:
|
||
|
REPORT_ERROR(MAJOR, E_INVALID_SELECTION, NO_MSG);
|
||
|
break;
|
||
|
}
|
||
|
FmPcdUnlock(p_FmPcd, intFlags);
|
||
|
|
||
|
return counterVal;
|
||
|
}
|
||
|
|
||
|
|
||
|
t_Error FmPcdPlcrCcGetSetParams(t_Handle h_FmPcd, uint16_t profileIndx ,uint32_t requiredAction)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd *)h_FmPcd;
|
||
|
t_FmPcdPlcr *p_FmPcdPlcr = p_FmPcd->p_FmPcdPlcr;
|
||
|
t_FmPcdPlcrRegs *p_FmPcdPlcrRegs = p_FmPcdPlcr->p_FmPcdPlcrRegs;
|
||
|
uint32_t tmpReg32, intFlags;
|
||
|
|
||
|
if (p_FmPcd->h_Hc)
|
||
|
return FmHcPcdPlcrCcGetSetParams(p_FmPcd->h_Hc, profileIndx, requiredAction);
|
||
|
|
||
|
if (profileIndx >= FM_PCD_PLCR_NUM_ENTRIES)
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_VALUE,("Policer profile out of range"));
|
||
|
|
||
|
if (FmPcdPlcrProfileTryLock(p_FmPcd, profileIndx, FALSE))
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_STATE,("Lock on PP FAILED"));
|
||
|
|
||
|
intFlags = FmPcdLock(p_FmPcd);
|
||
|
WritePar(p_FmPcd, FmPcdPlcrBuildReadPlcrActionReg(profileIndx));
|
||
|
|
||
|
if(!FmPcdPlcrHwProfileIsValid(GET_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pemode)))
|
||
|
{
|
||
|
FmPcdUnlock(p_FmPcd, intFlags);
|
||
|
RELEASE_LOCK(p_FmPcd->p_FmPcdPlcr->profiles[profileIndx].lock);
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_VALUE,("Policer profile is not valid"));
|
||
|
}
|
||
|
|
||
|
ASSERT_COND(p_FmPcd->p_FmPcdPlcr->profiles[profileIndx].valid);
|
||
|
|
||
|
if(!p_FmPcd->p_FmPcdPlcr->profiles[profileIndx].pointedOwners ||
|
||
|
!(p_FmPcd->p_FmPcdPlcr->profiles[profileIndx].requiredAction & requiredAction))
|
||
|
{
|
||
|
if(requiredAction & UPDATE_NIA_ENQ_WITHOUT_DMA)
|
||
|
{
|
||
|
if((p_FmPcd->p_FmPcdPlcr->profiles[profileIndx].nextEngineOnGreen!= e_FM_PCD_DONE) ||
|
||
|
(p_FmPcd->p_FmPcdPlcr->profiles[profileIndx].nextEngineOnYellow!= e_FM_PCD_DONE) ||
|
||
|
(p_FmPcd->p_FmPcdPlcr->profiles[profileIndx].nextEngineOnRed!= e_FM_PCD_DONE))
|
||
|
{
|
||
|
FmPcdUnlock(p_FmPcd, intFlags);
|
||
|
RETURN_ERROR (MAJOR, E_OK, ("In this case the next engine can be e_FM_PCD_DONE"));
|
||
|
}
|
||
|
|
||
|
if(p_FmPcd->p_FmPcdPlcr->profiles[profileIndx].paramsOnGreen.action == e_FM_PCD_ENQ_FRAME)
|
||
|
{
|
||
|
tmpReg32 = GET_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pegnia);
|
||
|
if(!(tmpReg32 & (NIA_ENG_BMI | NIA_BMI_AC_ENQ_FRAME)))
|
||
|
{
|
||
|
FmPcdUnlock(p_FmPcd, intFlags);
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_STATE, ("Next engine of this policer profile has to be assigned to FM_PCD_DONE"));
|
||
|
}
|
||
|
tmpReg32 |= NIA_BMI_AC_ENQ_FRAME_WITHOUT_DMA;
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pegnia, tmpReg32);
|
||
|
tmpReg32 = FmPcdPlcrBuildWritePlcrActionReg(profileIndx);
|
||
|
tmpReg32 |= FM_PCD_PLCR_PAR_PWSEL_PEGNIA;
|
||
|
WritePar(p_FmPcd, tmpReg32);
|
||
|
}
|
||
|
|
||
|
if(p_FmPcd->p_FmPcdPlcr->profiles[profileIndx].paramsOnYellow.action == e_FM_PCD_ENQ_FRAME)
|
||
|
{
|
||
|
tmpReg32 = GET_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_peynia);
|
||
|
if(!(tmpReg32 & (NIA_ENG_BMI | NIA_BMI_AC_ENQ_FRAME)))
|
||
|
{
|
||
|
FmPcdUnlock(p_FmPcd, intFlags);
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_STATE, ("Next engine of this policer profile has to be assigned to FM_PCD_DONE"));
|
||
|
}
|
||
|
tmpReg32 |= NIA_BMI_AC_ENQ_FRAME_WITHOUT_DMA;
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_peynia, tmpReg32);
|
||
|
tmpReg32 = FmPcdPlcrBuildWritePlcrActionReg(profileIndx);
|
||
|
tmpReg32 |= FM_PCD_PLCR_PAR_PWSEL_PEYNIA;
|
||
|
WritePar(p_FmPcd, tmpReg32);
|
||
|
}
|
||
|
|
||
|
if(p_FmPcd->p_FmPcdPlcr->profiles[profileIndx].paramsOnRed.action == e_FM_PCD_ENQ_FRAME)
|
||
|
{
|
||
|
tmpReg32 = GET_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pernia);
|
||
|
if(!(tmpReg32 & (NIA_ENG_BMI | NIA_BMI_AC_ENQ_FRAME)))
|
||
|
{
|
||
|
FmPcdUnlock(p_FmPcd, intFlags);
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_STATE, ("Next engine of this policer profile has to be assigned to FM_PCD_DONE"));
|
||
|
}
|
||
|
tmpReg32 |= NIA_BMI_AC_ENQ_FRAME_WITHOUT_DMA;
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pernia, tmpReg32);
|
||
|
tmpReg32 = FmPcdPlcrBuildWritePlcrActionReg(profileIndx);
|
||
|
tmpReg32 |= FM_PCD_PLCR_PAR_PWSEL_PERNIA;
|
||
|
WritePar(p_FmPcd, tmpReg32);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
FmPcdUnlock(p_FmPcd, intFlags);
|
||
|
|
||
|
p_FmPcd->p_FmPcdPlcr->profiles[profileIndx].pointedOwners += 1;
|
||
|
p_FmPcd->p_FmPcdPlcr->profiles[profileIndx].requiredAction |= requiredAction;
|
||
|
|
||
|
RELEASE_LOCK(p_FmPcd->p_FmPcdPlcr->profiles[profileIndx].lock);
|
||
|
|
||
|
return E_OK;
|
||
|
}
|
||
|
|
||
|
t_Error FM_PCD_PlcrSetProfileCounter(t_Handle h_FmPcd, t_Handle h_Profile, e_FmPcdPlcrProfileCounters counter, uint32_t value)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
uint16_t profileIndx = (uint16_t)(PTR_TO_UINT(h_Profile)-1);
|
||
|
t_FmPcdPlcrRegs *p_FmPcdPlcrRegs;
|
||
|
uint32_t tmpReg32, intFlags;
|
||
|
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd, E_INVALID_HANDLE);
|
||
|
SANITY_CHECK_RETURN_ERROR(!p_FmPcd->p_FmPcdDriverParam, E_INVALID_STATE);
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd->p_FmPcdPlcr, E_INVALID_HANDLE);
|
||
|
|
||
|
if (p_FmPcd->h_Hc)
|
||
|
return FmHcPcdPlcrSetProfileCounter(p_FmPcd->h_Hc, h_Profile, counter, value);
|
||
|
|
||
|
p_FmPcdPlcrRegs = p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs;
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcdPlcrRegs, E_INVALID_HANDLE);
|
||
|
|
||
|
intFlags = FmPcdLock(p_FmPcd);
|
||
|
switch (counter)
|
||
|
{
|
||
|
case e_FM_PCD_PLCR_PROFILE_GREEN_PACKET_TOTAL_COUNTER:
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_pegpc, value);
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_PROFILE_YELLOW_PACKET_TOTAL_COUNTER:
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_peypc, value);
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_PROFILE_RED_PACKET_TOTAL_COUNTER:
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_perpc, value);
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_PROFILE_RECOLOURED_YELLOW_PACKET_TOTAL_COUNTER:
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_perypc ,value);
|
||
|
break;
|
||
|
case e_FM_PCD_PLCR_PROFILE_RECOLOURED_RED_PACKET_TOTAL_COUNTER:
|
||
|
WRITE_UINT32(p_FmPcdPlcrRegs->profileRegs.fmpl_perrpc ,value);
|
||
|
break;
|
||
|
default:
|
||
|
RETURN_ERROR(MAJOR, E_INVALID_SELECTION, NO_MSG);
|
||
|
}
|
||
|
|
||
|
/* Activate the atomic write action by writing FMPL_PAR with: GO=1, RW=1, PSI=0, PNUM =
|
||
|
* Profile Number, PWSEL=0xFFFF (select all words).
|
||
|
*/
|
||
|
tmpReg32 = FmPcdPlcrBuildWritePlcrActionReg(profileIndx);
|
||
|
tmpReg32 |= FmPcdPlcrBuildCounterProfileReg(counter);
|
||
|
WritePar(p_FmPcd, tmpReg32);
|
||
|
FmPcdUnlock(p_FmPcd, intFlags);
|
||
|
|
||
|
return E_OK;
|
||
|
}
|
||
|
|
||
|
t_Error FM_PCD_ConfigPlcrNumOfSharedProfiles(t_Handle h_FmPcd, uint16_t numOfSharedPlcrProfiles)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd, E_INVALID_HANDLE);
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd->p_FmPcdDriverParam, E_INVALID_HANDLE);
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd->p_FmPcdPlcr, E_INVALID_HANDLE);
|
||
|
|
||
|
p_FmPcd->p_FmPcdPlcr->numOfSharedProfiles = numOfSharedPlcrProfiles;
|
||
|
|
||
|
return E_OK;
|
||
|
}
|
||
|
|
||
|
|
||
|
/* ... */
|
||
|
|
||
|
#if (defined(DEBUG_ERRORS) && (DEBUG_ERRORS > 0))
|
||
|
t_Error FM_PCD_PlcrDumpRegs(t_Handle h_FmPcd)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
int i = 0;
|
||
|
t_FmPcdIpcMsg msg;
|
||
|
|
||
|
DECLARE_DUMP;
|
||
|
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd, E_INVALID_HANDLE);
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd->p_FmPcdPlcr, E_INVALID_HANDLE);
|
||
|
SANITY_CHECK_RETURN_ERROR(!p_FmPcd->p_FmPcdDriverParam, E_INVALID_STATE);
|
||
|
|
||
|
if(p_FmPcd->guestId != NCSW_MASTER_ID)
|
||
|
{
|
||
|
memset(&msg, 0, sizeof(msg));
|
||
|
msg.msgId = FM_PCD_PLCR_DUMP_REGS;
|
||
|
return XX_IpcSendMessage(p_FmPcd->h_IpcSession,
|
||
|
(uint8_t*)&msg,
|
||
|
sizeof(msg.msgId),
|
||
|
NULL,
|
||
|
NULL,
|
||
|
NULL,
|
||
|
NULL);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
DUMP_SUBTITLE(("\n"));
|
||
|
DUMP_TITLE(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs, ("FmPcdPlcrRegs Regs"));
|
||
|
|
||
|
DUMP_VAR(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs,fmpl_gcr);
|
||
|
DUMP_VAR(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs,fmpl_gsr);
|
||
|
DUMP_VAR(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs,fmpl_evr);
|
||
|
DUMP_VAR(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs,fmpl_ier);
|
||
|
DUMP_VAR(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs,fmpl_ifr);
|
||
|
DUMP_VAR(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs,fmpl_eevr);
|
||
|
DUMP_VAR(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs,fmpl_eier);
|
||
|
DUMP_VAR(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs,fmpl_eifr);
|
||
|
DUMP_VAR(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs,fmpl_rpcnt);
|
||
|
DUMP_VAR(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs,fmpl_ypcnt);
|
||
|
DUMP_VAR(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs,fmpl_rrpcnt);
|
||
|
DUMP_VAR(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs,fmpl_rypcnt);
|
||
|
DUMP_VAR(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs,fmpl_tpcnt);
|
||
|
DUMP_VAR(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs,fmpl_flmcnt);
|
||
|
|
||
|
DUMP_VAR(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs,fmpl_serc);
|
||
|
DUMP_VAR(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs,fmpl_upcr);
|
||
|
DUMP_VAR(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs,fmpl_dpmr);
|
||
|
|
||
|
|
||
|
DUMP_TITLE(&p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs->fmpl_pmr, ("fmpl_pmr"));
|
||
|
DUMP_SUBSTRUCT_ARRAY(i, 63)
|
||
|
{
|
||
|
DUMP_MEMORY(&p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs->fmpl_pmr[i], sizeof(uint32_t));
|
||
|
}
|
||
|
|
||
|
return E_OK;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
t_Error FM_PCD_PlcrProfileDumpRegs(t_Handle h_FmPcd, t_Handle h_Profile)
|
||
|
{
|
||
|
t_FmPcd *p_FmPcd = (t_FmPcd*)h_FmPcd;
|
||
|
t_FmPcdPlcrInterModuleProfileRegs *p_ProfilesRegs;
|
||
|
uint32_t tmpReg, intFlags;
|
||
|
uint16_t profileIndx = (uint16_t)(PTR_TO_UINT(h_Profile)-1);
|
||
|
t_FmPcdIpcMsg msg;
|
||
|
|
||
|
DECLARE_DUMP;
|
||
|
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd, E_INVALID_HANDLE);
|
||
|
SANITY_CHECK_RETURN_ERROR(p_FmPcd->p_FmPcdPlcr, E_INVALID_HANDLE);
|
||
|
SANITY_CHECK_RETURN_ERROR(!p_FmPcd->p_FmPcdDriverParam, E_INVALID_STATE);
|
||
|
|
||
|
if(p_FmPcd->guestId != NCSW_MASTER_ID)
|
||
|
{
|
||
|
memset(&msg, 0, sizeof(msg));
|
||
|
msg.msgId = FM_PCD_PLCR_PROFILE_DUMP_REGS;
|
||
|
memcpy(msg.msgBody, (uint8_t *)&h_Profile, sizeof(uint32_t));
|
||
|
return XX_IpcSendMessage(p_FmPcd->h_IpcSession,
|
||
|
(uint8_t*)&msg,
|
||
|
sizeof(msg.msgId) + sizeof(uint32_t),
|
||
|
NULL,
|
||
|
NULL,
|
||
|
NULL,
|
||
|
NULL);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
DUMP_SUBTITLE(("\n"));
|
||
|
DUMP_TITLE(p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs, ("FmPcdPlcrRegs Profile Regs"));
|
||
|
|
||
|
p_ProfilesRegs = &p_FmPcd->p_FmPcdPlcr->p_FmPcdPlcrRegs->profileRegs;
|
||
|
|
||
|
tmpReg = FmPcdPlcrBuildReadPlcrActionReg((uint16_t)profileIndx);
|
||
|
intFlags = FmPcdLock(p_FmPcd);
|
||
|
WritePar(p_FmPcd, tmpReg);
|
||
|
|
||
|
DUMP_TITLE(p_ProfilesRegs, ("Profile %d regs", profileIndx));
|
||
|
|
||
|
DUMP_VAR(p_ProfilesRegs, fmpl_pemode);
|
||
|
DUMP_VAR(p_ProfilesRegs, fmpl_pegnia);
|
||
|
DUMP_VAR(p_ProfilesRegs, fmpl_peynia);
|
||
|
DUMP_VAR(p_ProfilesRegs, fmpl_pernia);
|
||
|
DUMP_VAR(p_ProfilesRegs, fmpl_pecir);
|
||
|
DUMP_VAR(p_ProfilesRegs, fmpl_pecbs);
|
||
|
DUMP_VAR(p_ProfilesRegs, fmpl_pepepir_eir);
|
||
|
DUMP_VAR(p_ProfilesRegs, fmpl_pepbs_ebs);
|
||
|
DUMP_VAR(p_ProfilesRegs, fmpl_pelts);
|
||
|
DUMP_VAR(p_ProfilesRegs, fmpl_pects);
|
||
|
DUMP_VAR(p_ProfilesRegs, fmpl_pepts_ets);
|
||
|
DUMP_VAR(p_ProfilesRegs, fmpl_pegpc);
|
||
|
DUMP_VAR(p_ProfilesRegs, fmpl_peypc);
|
||
|
DUMP_VAR(p_ProfilesRegs, fmpl_perpc);
|
||
|
DUMP_VAR(p_ProfilesRegs, fmpl_perypc);
|
||
|
DUMP_VAR(p_ProfilesRegs, fmpl_perrpc);
|
||
|
FmPcdUnlock(p_FmPcd, intFlags);
|
||
|
|
||
|
return E_OK;
|
||
|
}
|
||
|
}
|
||
|
#endif /* (defined(DEBUG_ERRORS) && ... */
|