You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
suricata/src/decode-esp.c

208 lines
4.9 KiB
C

/* Copyright (C) 2020-2021 Open Information Security Foundation
*
* You can copy, redistribute or modify this Program under the terms of
* the GNU General Public License version 2 as published by the Free
* Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* version 2 along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
* 02110-1301, USA.
*/
/**
* \ingroup decode
*
* @{
*/
/**
* \file
*
* Decode Encapsulating Security Payload (ESP)
*/
#include "suricata-common.h"
#include "decode-esp.h"
#include "flow.h"
#include "util-validate.h"
static int DecodeESPPacket(ThreadVars *tv, Packet *p, const uint8_t *pkt, uint16_t len)
{
DEBUG_VALIDATE_BUG_ON(pkt == NULL);
if (unlikely(len < ESP_HEADER_LEN)) {
ENGINE_SET_INVALID_EVENT(p, ESP_PKT_TOO_SMALL);
return -1;
}
p->esph = (ESPHdr *)pkt;
p->payload = (uint8_t *)pkt + sizeof(ESPHdr);
p->payload_len = len - sizeof(ESPHdr);
p->proto = IPPROTO_ESP;
return 0;
}
/**
* \brief Function to decode IPSEC-ESP packets
* \param tv thread vars
* \param dtv decoder thread vars
* \param p packet
* \param pkt raw packet data
* \param len length in bytes of pkt array
* \retval TM_ECODE_OK or TM_ECODE_FAILED on serious error
*/
int DecodeESP(ThreadVars *tv, DecodeThreadVars *dtv, Packet *p, const uint8_t *pkt, uint16_t len)
{
DEBUG_VALIDATE_BUG_ON(pkt == NULL);
StatsIncr(tv, dtv->counter_esp);
if (!PacketIncreaseCheckLayers(p)) {
return TM_ECODE_FAILED;
}
if (unlikely(DecodeESPPacket(tv, p, pkt, len) < 0)) {
CLEAR_ESP_PACKET(p);
return TM_ECODE_FAILED;
}
SCLogDebug("ESP spi: %" PRIu32 " sequence: %" PRIu32, ESP_GET_SPI(p), ESP_GET_SEQUENCE(p));
FlowSetupPacket(p);
return TM_ECODE_OK;
}
#ifdef UNITTESTS
#include "util-unittest.h"
/** \test Successful decoding */
static int DecodeESPTest01(void)
{
uint8_t raw_esp[] = { 0x00, 0x00, 0x00, 0x7b, 0x00, 0x00, 0x00, 0x08 };
Packet *p = PacketGetFromAlloc();
FAIL_IF_NULL(p);
ThreadVars tv;
DecodeThreadVars dtv;
memset(&tv, 0, sizeof(ThreadVars));
memset(&dtv, 0, sizeof(DecodeThreadVars));
int ret = DecodeESP(&tv, &dtv, p, raw_esp, sizeof(raw_esp));
FAIL_IF(ret != TM_ECODE_OK);
FAIL_IF(p->proto != IPPROTO_ESP);
FAIL_IF(p->payload_len != sizeof(raw_esp) - ESP_HEADER_LEN);
FAIL_IF(ESP_GET_SPI(p) != 0x7b);
FAIL_IF(ESP_GET_SEQUENCE(p) != 0x08);
SCFree(p);
PASS;
}
/** \test Successful decoding, with payload data */
static int DecodeESPTest02(void)
{
uint8_t raw_esp[] = { 0x00, 0x00, 0x00, 0x7b, 0x00, 0x00, 0x00, 0x08, 0xFF, 0xFF };
Packet *p = PacketGetFromAlloc();
FAIL_IF_NULL(p);
ThreadVars tv;
DecodeThreadVars dtv;
memset(&tv, 0, sizeof(ThreadVars));
memset(&dtv, 0, sizeof(DecodeThreadVars));
int ret = DecodeESP(&tv, &dtv, p, raw_esp, sizeof(raw_esp));
FAIL_IF(ret != TM_ECODE_OK);
FAIL_IF(p->proto != IPPROTO_ESP);
FAIL_IF(p->payload_len != sizeof(raw_esp) - ESP_HEADER_LEN);
FAIL_IF(memcmp(p->payload, raw_esp + ESP_HEADER_LEN, p->payload_len) != 0);
FAIL_IF(ESP_GET_SPI(p) != 0x7b);
FAIL_IF(ESP_GET_SEQUENCE(p) != 0x08);
SCFree(p);
PASS;
}
/** \test Failure decoding, not enough data */
static int DecodeESPTest03(void)
{
uint8_t raw_esp[] = { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 };
Packet *p = PacketGetFromAlloc();
FAIL_IF_NULL(p);
ThreadVars tv;
DecodeThreadVars dtv;
memset(&tv, 0, sizeof(ThreadVars));
memset(&dtv, 0, sizeof(DecodeThreadVars));
int ret = DecodeESP(&tv, &dtv, p, raw_esp, sizeof(raw_esp));
FAIL_IF(ret != TM_ECODE_FAILED);
// expect ESP_PKT_TOO_SMALL
FAIL_IF_NOT(ENGINE_ISSET_EVENT(p, ESP_PKT_TOO_SMALL));
SCFree(p);
PASS;
}
/** \test Failure decoding, no data */
static int DecodeESPTest04(void)
{
uint8_t raw_esp[] = {};
Packet *p = PacketGetFromAlloc();
FAIL_IF_NULL(p);
ThreadVars tv;
DecodeThreadVars dtv;
memset(&tv, 0, sizeof(ThreadVars));
memset(&dtv, 0, sizeof(DecodeThreadVars));
int ret = DecodeESP(&tv, &dtv, p, raw_esp, sizeof(raw_esp));
FAIL_IF(ret != TM_ECODE_FAILED);
// expect ESP_PKT_TOO_SMALL
FAIL_IF_NOT(ENGINE_ISSET_EVENT(p, ESP_PKT_TOO_SMALL));
SCFree(p);
PASS;
}
#endif /* UNITTESTS */
void DecodeESPRegisterTests(void)
{
#ifdef UNITTESTS
UtRegisterTest("DecodeESPTest01", DecodeESPTest01);
UtRegisterTest("DecodeESPTest02", DecodeESPTest02);
UtRegisterTest("DecodeESPTest03", DecodeESPTest03);
UtRegisterTest("DecodeESPTest04", DecodeESPTest04);
#endif /* UNITTESTS */
}
/**
* @}
*/