1 /*
2 * FreeRTOS+TCP <DEVELOPMENT BRANCH>
3 * Copyright (C) 2022 Amazon.com, Inc. or its affiliates. All Rights Reserved.
4 *
5 * SPDX-License-Identifier: MIT
6 *
7 * Permission is hereby granted, free of charge, to any person obtaining a copy of
8 * this software and associated documentation files (the "Software"), to deal in
9 * the Software without restriction, including without limitation the rights to
10 * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
11 * the Software, and to permit persons to whom the Software is furnished to do so,
12 * subject to the following conditions:
13 *
14 * The above copyright notice and this permission notice shall be included in all
15 * copies or substantial portions of the Software.
16 *
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
19 * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
20 * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
21 * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
22 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
23 *
24 * http://aws.amazon.com/freertos
25 * http://www.FreeRTOS.org
26 */
27
28
29 /* Include Unity header */
30 #include "unity.h"
31
32 /* Include standard libraries */
33 #include <stdlib.h>
34 #include <string.h>
35 #include <stdint.h>
36
37 #include "FreeRTOSIPConfig.h"
38
39 #include "FreeRTOS_IPv4_Utils.h"
40
41 #include "catch_assert.h"
42
43 /* ============================== Test Cases ============================== */
44
45 /**
46 * @brief test_vSetMultiCastIPv4MacAddress
47 * To validate if vSetMultiCastIPv4MacAddress sets IPv4 multicast MAC address correctly
48 * into input pointer.
49 */
test_vSetMultiCastIPv4MacAddress(void)50 void test_vSetMultiCastIPv4MacAddress( void )
51 {
52 uint32_t ulIP = 0xABCDEF12;
53 MACAddress_t xMACAddress;
54
55 vSetMultiCastIPv4MacAddress( ulIP, &xMACAddress );
56
57 TEST_ASSERT_EQUAL( ( uint8_t ) 0x01U, xMACAddress.ucBytes[ 0 ] );
58 TEST_ASSERT_EQUAL( ( uint8_t ) 0x00U, xMACAddress.ucBytes[ 1 ] );
59 TEST_ASSERT_EQUAL( ( uint8_t ) 0x5EU, xMACAddress.ucBytes[ 2 ] );
60 TEST_ASSERT_EQUAL( ( uint8_t ) ( ( FreeRTOS_ntohl( ulIP ) >> 16 ) & 0x7fU ), xMACAddress.ucBytes[ 3 ] );
61 TEST_ASSERT_EQUAL( ( uint8_t ) ( ( FreeRTOS_ntohl( ulIP ) >> 8 ) & 0xffU ), xMACAddress.ucBytes[ 4 ] );
62 TEST_ASSERT_EQUAL( ( uint8_t ) ( ( FreeRTOS_ntohl( ulIP ) ) & 0xffU ), xMACAddress.ucBytes[ 5 ] );
63 }
64
65 /**
66 * @brief test_prvChecksumIPv4Checks_IPLengthLessThanHeaderLength
67 * To validate if prvChecksumIPv4Checks sets ipINVALID_LENGTH in checksum field when
68 * length in IPv4 header is less than ucVersionHeaderLength.
69 */
test_prvChecksumIPv4Checks_IPLengthLessThanHeaderLength()70 void test_prvChecksumIPv4Checks_IPLengthLessThanHeaderLength()
71 {
72 BaseType_t xReturn;
73 IPPacket_t xIPPacket;
74 struct xPacketSummary xSet;
75 uint8_t ucVersionHeaderLength = 20;
76 uint16_t usLength = ucVersionHeaderLength - 1;
77 uint8_t pucEthernetBuffer[ ipconfigTCP_MSS ];
78 size_t uxBufferLength = usLength + ipSIZE_OF_ETH_HEADER;
79
80 memset( &xIPPacket, 0, sizeof( xIPPacket ) );
81 memset( &xSet, 0, sizeof( xSet ) );
82
83 xSet.pxIPPacket = &xIPPacket;
84 xIPPacket.xIPHeader.usLength = FreeRTOS_htons( usLength );
85 xIPPacket.xIPHeader.ucVersionHeaderLength = ( ucVersionHeaderLength >> 2 );
86
87 xReturn = prvChecksumIPv4Checks( pucEthernetBuffer, uxBufferLength, &xSet );
88
89 TEST_ASSERT_EQUAL( 3, xReturn );
90 TEST_ASSERT_EQUAL( ipINVALID_LENGTH, xSet.usChecksum );
91 }
92
93 /**
94 * @brief test_prvChecksumIPv4Checks_BufferLessIPPacket
95 * To validate if prvChecksumIPv4Checks sets ipINVALID_LENGTH in checksum field when
96 * buffer size is less than IPv4 packet minimum requirement.
97 */
test_prvChecksumIPv4Checks_BufferLessIPPacket()98 void test_prvChecksumIPv4Checks_BufferLessIPPacket()
99 {
100 BaseType_t xReturn;
101 IPPacket_t xIPPacket;
102 struct xPacketSummary xSet;
103 uint8_t ucVersionHeaderLength = 20;
104 uint16_t usLength = ucVersionHeaderLength;
105 uint8_t pucEthernetBuffer[ ipconfigTCP_MSS ];
106 size_t uxBufferLength = ipSIZE_OF_ETH_HEADER + ipSIZE_OF_IPv4_HEADER - 1;
107
108 memset( &xIPPacket, 0, sizeof( xIPPacket ) );
109 memset( &xSet, 0, sizeof( xSet ) );
110
111 xSet.pxIPPacket = &xIPPacket;
112 xIPPacket.xIPHeader.usLength = FreeRTOS_htons( usLength );
113 xIPPacket.xIPHeader.ucVersionHeaderLength = ( ucVersionHeaderLength >> 2 );
114
115 xReturn = prvChecksumIPv4Checks( pucEthernetBuffer, uxBufferLength, &xSet );
116
117 TEST_ASSERT_EQUAL( 4, xReturn );
118 TEST_ASSERT_EQUAL( ipINVALID_LENGTH, xSet.usChecksum );
119 }
120
121 /**
122 * @brief test_prvChecksumIPv4Checks_BufferLessIPHeaderLength
123 * To validate if prvChecksumIPv4Checks sets ipINVALID_LENGTH in checksum field when
124 * buffer size is less than IPv4 header length.
125 */
test_prvChecksumIPv4Checks_BufferLessIPHeaderLength()126 void test_prvChecksumIPv4Checks_BufferLessIPHeaderLength()
127 {
128 BaseType_t xReturn;
129 IPPacket_t xIPPacket;
130 struct xPacketSummary xSet;
131 uint8_t ucVersionHeaderLength = 24;
132 uint16_t usLength = ucVersionHeaderLength;
133 uint8_t pucEthernetBuffer[ ipconfigTCP_MSS ];
134 size_t uxBufferLength = ipSIZE_OF_ETH_HEADER + ucVersionHeaderLength - 1;
135
136 memset( &xIPPacket, 0, sizeof( xIPPacket ) );
137 memset( &xSet, 0, sizeof( xSet ) );
138
139 xSet.pxIPPacket = &xIPPacket;
140 xIPPacket.xIPHeader.usLength = FreeRTOS_htons( usLength );
141 xIPPacket.xIPHeader.ucVersionHeaderLength = ( ucVersionHeaderLength >> 2 );
142
143 xReturn = prvChecksumIPv4Checks( pucEthernetBuffer, uxBufferLength, &xSet );
144
145 TEST_ASSERT_EQUAL( 5, xReturn );
146 TEST_ASSERT_EQUAL( ipINVALID_LENGTH, xSet.usChecksum );
147 }
148
149 /**
150 * @brief test_prvChecksumIPv4Checks_BufferLessIPPayloadLength
151 * To validate if prvChecksumIPv4Checks sets ipINVALID_LENGTH in checksum field when
152 * buffer size is less than IPv4 header payload length.
153 */
test_prvChecksumIPv4Checks_BufferLessIPPayloadLength()154 void test_prvChecksumIPv4Checks_BufferLessIPPayloadLength()
155 {
156 BaseType_t xReturn;
157 IPPacket_t xIPPacket;
158 struct xPacketSummary xSet;
159 uint8_t ucVersionHeaderLength = 20;
160 uint16_t usLength = ucVersionHeaderLength + ipSIZE_OF_TCP_HEADER;
161 uint8_t pucEthernetBuffer[ ipconfigTCP_MSS ];
162 size_t uxBufferLength = ipSIZE_OF_ETH_HEADER + usLength - 1;
163
164 memset( &xIPPacket, 0, sizeof( xIPPacket ) );
165 memset( &xSet, 0, sizeof( xSet ) );
166
167 xSet.pxIPPacket = &xIPPacket;
168 xIPPacket.xIPHeader.usLength = FreeRTOS_htons( usLength );
169 xIPPacket.xIPHeader.ucVersionHeaderLength = ( ucVersionHeaderLength >> 2 );
170
171 xReturn = prvChecksumIPv4Checks( pucEthernetBuffer, uxBufferLength, &xSet );
172
173 TEST_ASSERT_EQUAL( 6, xReturn );
174 TEST_ASSERT_EQUAL( ipINVALID_LENGTH, xSet.usChecksum );
175 }
176
177 /**
178 * @brief test_prvChecksumIPv4Checks_Pass
179 * To validate if prvChecksumIPv4Checks returns 0 when pass.
180 */
test_prvChecksumIPv4Checks_Pass()181 void test_prvChecksumIPv4Checks_Pass()
182 {
183 BaseType_t xReturn;
184 IPPacket_t xIPPacket;
185 struct xPacketSummary xSet;
186 uint8_t ucVersionHeaderLength = 20;
187 uint16_t usLength = ucVersionHeaderLength + ipSIZE_OF_TCP_HEADER;
188 uint8_t pucEthernetBuffer[ ipconfigTCP_MSS ];
189 size_t uxBufferLength = ipSIZE_OF_ETH_HEADER + usLength;
190
191 memset( &xIPPacket, 0, sizeof( xIPPacket ) );
192 memset( &xSet, 0, sizeof( xSet ) );
193
194 xSet.pxIPPacket = &xIPPacket;
195 xIPPacket.xIPHeader.usLength = FreeRTOS_htons( usLength );
196 xIPPacket.xIPHeader.ucVersionHeaderLength = ( ucVersionHeaderLength >> 2 );
197 xIPPacket.xIPHeader.ucProtocol = ipPROTOCOL_TCP;
198
199 xReturn = prvChecksumIPv4Checks( pucEthernetBuffer, uxBufferLength, &xSet );
200
201 TEST_ASSERT_EQUAL( 0, xReturn );
202 TEST_ASSERT_EQUAL( ipPROTOCOL_TCP, xSet.ucProtocol );
203 TEST_ASSERT_EQUAL( &pucEthernetBuffer[ ipSIZE_OF_ETH_HEADER + ucVersionHeaderLength ], xSet.pxProtocolHeaders );
204 TEST_ASSERT_EQUAL( ipSIZE_OF_TCP_HEADER, xSet.usProtocolBytes );
205 }
206