From 0a0b9cb656d03ce2b60579e099b2185c2bb095f9 Mon Sep 17 00:00:00 2001 From: Jacob Hageman Date: Tue, 4 May 2021 13:36:28 +0000 Subject: [PATCH] Fix #1419, Resolve sequence count auto-increment rollover bug --- modules/core_api/fsw/inc/cfe_msg.h | 16 ++++++++++++ modules/core_api/ut-stubs/src/ut_msg_stubs.c | 12 +++++++++ modules/msg/fsw/src/cfe_msg_ccsdspri.c | 15 +++++++++++ .../msg/ut-coverage/test_cfe_msg_ccsdspri.c | 19 ++++++++++---- modules/sb/ut-coverage/sb_UT.c | 25 ++++++++++++++++--- modules/sbr/fsw/src/cfe_sbr_route_unsorted.c | 5 +++- .../ut-coverage/test_cfe_sbr_route_unsorted.c | 21 ++++++++++------ 7 files changed, 95 insertions(+), 18 deletions(-) diff --git a/modules/core_api/fsw/inc/cfe_msg.h b/modules/core_api/fsw/inc/cfe_msg.h index 22a40bddc..2ce5542f2 100644 --- a/modules/core_api/fsw/inc/cfe_msg.h +++ b/modules/core_api/fsw/inc/cfe_msg.h @@ -290,6 +290,22 @@ CFE_Status_t CFE_MSG_GetSequenceCount(const CFE_MSG_Message_t *MsgPtr, CFE_MSG_S */ CFE_Status_t CFE_MSG_SetSequenceCount(CFE_MSG_Message_t *MsgPtr, CFE_MSG_SequenceCount_t SeqCnt); +/*****************************************************************************/ +/** + * \brief Gets the next sequence count value (rolls over if appropriate) + * + * \par Description + * Abstract method to get the next valid sequence count value. + * Will roll over to zero for any input value greater than or + * equal to the maximum possible sequence count value given + * the field in the header. + * + * \param[in] SeqCnt Sequence count + * + * \return The next valid sequence count value + */ +CFE_MSG_SequenceCount_t CFE_MSG_GetNextSequenceCount(CFE_MSG_SequenceCount_t SeqCnt); + /*****************************************************************************/ /** * \brief Gets the message EDS version diff --git a/modules/core_api/ut-stubs/src/ut_msg_stubs.c b/modules/core_api/ut-stubs/src/ut_msg_stubs.c index c7d4cc8a0..80e4102d3 100644 --- a/modules/core_api/ut-stubs/src/ut_msg_stubs.c +++ b/modules/core_api/ut-stubs/src/ut_msg_stubs.c @@ -703,3 +703,15 @@ int32 CFE_MSG_ValidateChecksum(const CFE_MSG_Message_t *MsgPtr, bool *IsValid) return status; } + +/* + * ----------------------------------------------------------- + * Stub implementation of CFE_MSG_GetNextSequenceCount + * ----------------------------------------------------------- + */ +CFE_MSG_SequenceCount_t CFE_MSG_GetNextSequenceCount(CFE_MSG_SequenceCount_t SeqCnt) +{ + UT_Stub_RegisterContextGenericArg(UT_KEY(CFE_MSG_GetNextSequenceCount), SeqCnt); + + return UT_DEFAULT_IMPL(CFE_MSG_GetNextSequenceCount); +} diff --git a/modules/msg/fsw/src/cfe_msg_ccsdspri.c b/modules/msg/fsw/src/cfe_msg_ccsdspri.c index 780b93d23..92284b2b6 100644 --- a/modules/msg/fsw/src/cfe_msg_ccsdspri.c +++ b/modules/msg/fsw/src/cfe_msg_ccsdspri.c @@ -316,6 +316,21 @@ int32 CFE_MSG_SetSequenceCount(CFE_MSG_Message_t *MsgPtr, CFE_MSG_SequenceCount_ return CFE_SUCCESS; } +/****************************************************************************** + * Get next sequence count - See API and header file for details + */ +CFE_MSG_SequenceCount_t CFE_MSG_GetNextSequenceCount(CFE_MSG_SequenceCount_t SeqCnt) +{ + SeqCnt++; + + if (SeqCnt > CFE_MSG_SEQCNT_MASK) + { + SeqCnt = 0; + } + + return SeqCnt; +} + /****************************************************************************** * Get message size - See API and header file for details */ diff --git a/modules/msg/ut-coverage/test_cfe_msg_ccsdspri.c b/modules/msg/ut-coverage/test_cfe_msg_ccsdspri.c index 816ed4f83..a9d786f2b 100644 --- a/modules/msg/ut-coverage/test_cfe_msg_ccsdspri.c +++ b/modules/msg/ut-coverage/test_cfe_msg_ccsdspri.c @@ -396,10 +396,13 @@ void Test_MSG_SegmentationFlag(void) void Test_MSG_SequenceCount(void) { - CFE_MSG_Message_t msg; - CFE_MSG_ApId_t input[] = {0, TEST_SEQUENCE_MAX / 2, TEST_SEQUENCE_MAX}; - CFE_MSG_ApId_t actual = TEST_SEQUENCE_MAX; - int i; + CFE_MSG_Message_t msg; + const CFE_MSG_SequenceCount_t input[] = {0, TEST_SEQUENCE_MAX / 2, TEST_SEQUENCE_MAX}; + CFE_MSG_SequenceCount_t actual = TEST_SEQUENCE_MAX; + CFE_MSG_SequenceCount_t maxsc; + int i; + + memset(&maxsc, 0xFF, sizeof(maxsc)); UtPrintf("Bad parameter tests, Null pointers and invalid (max valid + 1, max)"); memset(&msg, 0, sizeof(msg)); @@ -410,7 +413,7 @@ void Test_MSG_SequenceCount(void) ASSERT_EQ(CFE_MSG_SetSequenceCount(NULL, input[0]), CFE_MSG_BAD_ARGUMENT); ASSERT_EQ(CFE_MSG_SetSequenceCount(&msg, TEST_SEQUENCE_MAX + 1), CFE_MSG_BAD_ARGUMENT); ASSERT_EQ(Test_MSG_NotZero(&msg), 0); - ASSERT_EQ(CFE_MSG_SetSequenceCount(&msg, 0xFFFF), CFE_MSG_BAD_ARGUMENT); + ASSERT_EQ(CFE_MSG_SetSequenceCount(&msg, maxsc), CFE_MSG_BAD_ARGUMENT); ASSERT_EQ(Test_MSG_NotZero(&msg), 0); UtPrintf("Set to all F's, various valid inputs"); @@ -452,6 +455,12 @@ void Test_MSG_SequenceCount(void) ASSERT_EQ(Test_MSG_NotZero(&msg), MSG_SEQUENCE_FLAG); } } + + UtPrintf("Fully exercise getting next sequence count"); + ASSERT_EQ(CFE_MSG_GetNextSequenceCount(0), 1); + ASSERT_EQ(CFE_MSG_GetNextSequenceCount(TEST_SEQUENCE_MAX / 2), (TEST_SEQUENCE_MAX / 2) + 1); + ASSERT_EQ(CFE_MSG_GetNextSequenceCount(TEST_SEQUENCE_MAX), 0); + ASSERT_EQ(CFE_MSG_GetNextSequenceCount(maxsc), 0); } /* diff --git a/modules/sb/ut-coverage/sb_UT.c b/modules/sb/ut-coverage/sb_UT.c index 3238b9752..888ed68f2 100644 --- a/modules/sb/ut-coverage/sb_UT.c +++ b/modules/sb/ut-coverage/sb_UT.c @@ -2820,7 +2820,7 @@ void Test_TransmitMsg_BasicSend(void) } /* end Test_TransmitMsg_BasicSend */ -/* Sequence count hook */ +/* Set sequence count hook */ static int32 UT_CheckSetSequenceCount(void *UserObj, int32 StubRetcode, uint32 CallCount, const UT_StubContext_t *Context) { @@ -2843,6 +2843,7 @@ void Test_TransmitMsg_SequenceCount(void) CFE_MSG_Type_t Type = CFE_MSG_Type_Tlm; uint32 PipeDepth = 10; CFE_MSG_SequenceCount_t SeqCnt; + CFE_MSG_SequenceCount_t SeqCntExpected; /* Set up hook for checking CFE_MSG_SetSequenceCount calls */ UT_SetHookFunction(UT_KEY(CFE_MSG_SetSequenceCount), UT_CheckSetSequenceCount, &SeqCnt); @@ -2850,12 +2851,17 @@ void Test_TransmitMsg_SequenceCount(void) SETUP(CFE_SB_CreatePipe(&PipeId, PipeDepth, "SeqCntTestPipe")); SETUP(CFE_SB_Subscribe(MsgId, PipeId)); + /* Note the Sequence count value doesn't really matter, just set unique to confirm use */ + SeqCntExpected = 1; + UT_SetDefaultReturnValue(UT_KEY(CFE_MSG_GetNextSequenceCount), SeqCntExpected); UT_SetDataBuffer(UT_KEY(CFE_MSG_GetMsgId), &MsgId, sizeof(MsgId), false); UT_SetDataBuffer(UT_KEY(CFE_MSG_GetSize), &Size, sizeof(Size), false); UT_SetDataBuffer(UT_KEY(CFE_MSG_GetType), &Type, sizeof(Type), false); + SETUP(CFE_SB_TransmitMsg(&TlmPkt.Hdr.Msg, true)); ASSERT_EQ(UT_GetStubCount(UT_KEY(CFE_MSG_SetSequenceCount)), 1); - ASSERT_EQ(SeqCnt, 1); + ASSERT_EQ(UT_GetStubCount(UT_KEY(CFE_MSG_GetNextSequenceCount)), 1); + ASSERT_EQ(SeqCnt, SeqCntExpected); UT_SetDataBuffer(UT_KEY(CFE_MSG_GetMsgId), &MsgId, sizeof(MsgId), false); UT_SetDataBuffer(UT_KEY(CFE_MSG_GetSize), &Size, sizeof(Size), false); @@ -2863,34 +2869,44 @@ void Test_TransmitMsg_SequenceCount(void) ASSERT(CFE_SB_TransmitMsg(&TlmPkt.Hdr.Msg, false)); /* Assert sequence count wasn't set */ + ASSERT_EQ(UT_GetStubCount(UT_KEY(CFE_MSG_GetNextSequenceCount)), 1); ASSERT_EQ(UT_GetStubCount(UT_KEY(CFE_MSG_SetSequenceCount)), 1); + SeqCntExpected = 2; + UT_SetDefaultReturnValue(UT_KEY(CFE_MSG_GetNextSequenceCount), SeqCntExpected); UT_SetDataBuffer(UT_KEY(CFE_MSG_GetMsgId), &MsgId, sizeof(MsgId), false); UT_SetDataBuffer(UT_KEY(CFE_MSG_GetSize), &Size, sizeof(Size), false); UT_SetDataBuffer(UT_KEY(CFE_MSG_GetType), &Type, sizeof(Type), false); ASSERT(CFE_SB_TransmitMsg(&TlmPkt.Hdr.Msg, true)); - ASSERT_EQ(SeqCnt, 2); + ASSERT_EQ(SeqCnt, SeqCntExpected); ASSERT_EQ(UT_GetStubCount(UT_KEY(CFE_MSG_SetSequenceCount)), 2); + ASSERT_EQ(UT_GetStubCount(UT_KEY(CFE_MSG_GetNextSequenceCount)), 2); EVTCNT(2); EVTSENT(CFE_SB_SUBSCRIPTION_RCVD_EID); SETUP(CFE_SB_Unsubscribe(MsgId, PipeId)); /* should have no subscribers now */ + SeqCntExpected = 3; + UT_SetDefaultReturnValue(UT_KEY(CFE_MSG_GetNextSequenceCount), SeqCntExpected); UT_SetDataBuffer(UT_KEY(CFE_MSG_GetMsgId), &MsgId, sizeof(MsgId), false); UT_SetDataBuffer(UT_KEY(CFE_MSG_GetSize), &Size, sizeof(Size), false); UT_SetDataBuffer(UT_KEY(CFE_MSG_GetType), &Type, sizeof(Type), false); SETUP(CFE_SB_TransmitMsg(&TlmPkt.Hdr.Msg, true)); /* increment to 3 */ ASSERT_EQ(UT_GetStubCount(UT_KEY(CFE_MSG_SetSequenceCount)), 3); + ASSERT_EQ(UT_GetStubCount(UT_KEY(CFE_MSG_GetNextSequenceCount)), 3); SETUP(CFE_SB_Subscribe(MsgId, PipeId)); /* resubscribe so we can receive a msg */ + SeqCntExpected = 4; + UT_SetDefaultReturnValue(UT_KEY(CFE_MSG_GetNextSequenceCount), SeqCntExpected); UT_SetDataBuffer(UT_KEY(CFE_MSG_GetMsgId), &MsgId, sizeof(MsgId), false); UT_SetDataBuffer(UT_KEY(CFE_MSG_GetSize), &Size, sizeof(Size), false); UT_SetDataBuffer(UT_KEY(CFE_MSG_GetType), &Type, sizeof(Type), false); SETUP(CFE_SB_TransmitMsg(&TlmPkt.Hdr.Msg, true)); /* increment to 4 */ - ASSERT_EQ(SeqCnt, 4); + ASSERT_EQ(SeqCnt, SeqCntExpected); ASSERT_EQ(UT_GetStubCount(UT_KEY(CFE_MSG_SetSequenceCount)), 4); + ASSERT_EQ(UT_GetStubCount(UT_KEY(CFE_MSG_GetNextSequenceCount)), 4); TEARDOWN(CFE_SB_DeletePipe(PipeId)); @@ -3135,6 +3151,7 @@ void Test_TransmitBuffer_IncrementSeqCnt(void) UtAssert_Failed("Unexpected NULL pointer returned from ZeroCopyGetPtr"); } + UT_SetDefaultReturnValue(UT_KEY(CFE_MSG_GetNextSequenceCount), 1); UT_SetDataBuffer(UT_KEY(CFE_MSG_GetMsgId), &MsgId, sizeof(MsgId), false); UT_SetDataBuffer(UT_KEY(CFE_MSG_GetSize), &Size, sizeof(Size), false); UT_SetDataBuffer(UT_KEY(CFE_MSG_GetType), &Type, sizeof(Type), false); diff --git a/modules/sbr/fsw/src/cfe_sbr_route_unsorted.c b/modules/sbr/fsw/src/cfe_sbr_route_unsorted.c index 40404fc53..d59bcc2db 100644 --- a/modules/sbr/fsw/src/cfe_sbr_route_unsorted.c +++ b/modules/sbr/fsw/src/cfe_sbr_route_unsorted.c @@ -38,6 +38,7 @@ #include #include "cfe_sb.h" +#include "cfe_msg.h" /****************************************************************************** * Type Definitions @@ -158,9 +159,11 @@ void CFE_SBR_SetDestListHeadPtr(CFE_SBR_RouteId_t RouteId, CFE_SB_DestinationD_t */ void CFE_SBR_IncrementSequenceCounter(CFE_SBR_RouteId_t RouteId) { + CFE_MSG_SequenceCount_t *cnt = &CFE_SBR_RDATA.RoutingTbl[CFE_SBR_RouteIdToValue(RouteId)].SeqCnt; + if (CFE_SBR_IsValidRouteId(RouteId)) { - CFE_SBR_RDATA.RoutingTbl[CFE_SBR_RouteIdToValue(RouteId)].SeqCnt++; + *cnt = CFE_MSG_GetNextSequenceCount(*cnt); } } diff --git a/modules/sbr/ut-coverage/test_cfe_sbr_route_unsorted.c b/modules/sbr/ut-coverage/test_cfe_sbr_route_unsorted.c index 22f436664..4e006bd0b 100644 --- a/modules/sbr/ut-coverage/test_cfe_sbr_route_unsorted.c +++ b/modules/sbr/ut-coverage/test_cfe_sbr_route_unsorted.c @@ -115,12 +115,13 @@ void Test_SBR_Route_Unsort_General(void) void Test_SBR_Route_Unsort_GetSet(void) { - CFE_SB_RouteId_Atom_t routeidx; - CFE_SB_MsgId_t msgid[3]; - CFE_SBR_RouteId_t routeid[3]; - CFE_SB_DestinationD_t dest[2]; - uint32 count; - uint32 i; + CFE_SB_RouteId_Atom_t routeidx; + CFE_SB_MsgId_t msgid[3]; + CFE_SBR_RouteId_t routeid[3]; + CFE_SB_DestinationD_t dest[2]; + CFE_MSG_SequenceCount_t seqcntexpected[] = {1, 2}; + uint32 count; + uint32 i; UtPrintf("Invalid route ID checks"); routeid[0] = CFE_SBR_INVALID_ROUTE_ID; @@ -167,20 +168,24 @@ void Test_SBR_Route_Unsort_GetSet(void) } /* Check the msgid matches and increment a sequence counter */ + UT_SetDefaultReturnValue(UT_KEY(CFE_MSG_GetNextSequenceCount), seqcntexpected[0]); for (i = 0; i < 3; i++) { ASSERT_TRUE(CFE_SB_MsgId_Equal(msgid[i], CFE_SBR_GetMsgId(routeid[i]))); CFE_SBR_IncrementSequenceCounter(routeid[0]); } + ASSERT_EQ(UT_GetStubCount(UT_KEY(CFE_MSG_GetNextSequenceCount)), 3); /* Increment route 1 once and set dest pointers */ + UT_SetDefaultReturnValue(UT_KEY(CFE_MSG_GetNextSequenceCount), seqcntexpected[1]); CFE_SBR_IncrementSequenceCounter(routeid[1]); + ASSERT_EQ(UT_GetStubCount(UT_KEY(CFE_MSG_GetNextSequenceCount)), 4); CFE_SBR_SetDestListHeadPtr(routeid[1], &dest[1]); CFE_SBR_SetDestListHeadPtr(routeid[2], &dest[0]); UtPrintf("Verify remaining set values"); - ASSERT_EQ(CFE_SBR_GetSequenceCounter(routeid[0]), 3); - ASSERT_EQ(CFE_SBR_GetSequenceCounter(routeid[1]), 1); + ASSERT_EQ(CFE_SBR_GetSequenceCounter(routeid[0]), seqcntexpected[0]); + ASSERT_EQ(CFE_SBR_GetSequenceCounter(routeid[1]), seqcntexpected[1]); ASSERT_EQ(CFE_SBR_GetSequenceCounter(routeid[2]), 0); UtAssert_ADDRESS_EQ(CFE_SBR_GetDestListHeadPtr(routeid[0]), NULL); UtAssert_ADDRESS_EQ(CFE_SBR_GetDestListHeadPtr(routeid[1]), &dest[1]);