*
* contains pointers and counters to store current state
*/
-typedef struct private_parser_context_s private_parser_context_t;
+typedef struct private_parser_s private_parser_t;
-struct private_parser_context_s {
+struct private_parser_s {
/**
* Public members
*/
- parser_context_t public;
+ parser_t public;
/**
* Current bit for reading in input data
*/
u_int8_t *input_roof;
-
-};
-
-/**
- * implementation of parser_context_t.destroy
- */
-static status_t parser_context_destroy(private_parser_context_t *this)
-{
- allocator_free(this);
-
- return SUCCESS;
-}
-
-
-/**
- * @brief Private data of a parser_t object
- */
-typedef struct private_parser_s private_parser_t;
-
-struct private_parser_s {
- /**
- * Public part of a generator object
- */
- parser_t public;
-
- /**
- * list of payloads and their description
- */
- payload_info_t **payload_infos;
-
/**
* logger object
*/
logger_t *logger;
-};
-
-/**
- * implementation of parser_t.create_context
- */
-static private_parser_context_t *create_context(private_parser_t *this, chunk_t data)
-{
- private_parser_context_t *context = allocator_alloc_thing(private_parser_context_t);
- if (this == NULL)
- {
- return NULL;
- }
-
- context->public.destroy = (status_t(*)(parser_context_t*)) parser_context_destroy;
- context->input = data.ptr;
- context->byte_pos = data.ptr;
- context->bit_pos = 0;
- context->input_roof = data.ptr + data.len;
- return context;
-}
+};
/**
* implementation of parser_context_t.parse_payload
*/
-static status_t parse_payload(private_parser_t *this, payload_type_t payload_type, void **data_struct, private_parser_context_t *context)
+static status_t parse_payload(private_parser_t *this, payload_type_t payload_type, payload_t **payload)
{
- payload_info_t *payload_info = NULL;
this->logger->log(this->logger, CONTROL, "Parsing a %s payload", mapping_find(payload_type_t_mappings, payload_type));
/* find payload in null terminated list*/
- payload_info = *(this->payload_infos);
- while (payload_info)
+
+ payload_t *pld;
+ void *output;
+ int current;
+ encoding_rule_t *rule;
+ size_t rule_count;
+
+ /* ok, do the parsing */
+ pld = payload_create(payload_type);
+ if (pld == NULL)
{
- if (payload_info->payload_type == payload_type)
+ this->logger->log(this->logger, ERROR, "Payload not supported");
+ return NOT_SUPPORTED;
+ }
+
+ /* base pointer for output, avoids casting in every rule */
+ output = pld;
+
+ pld->get_encoding_rules(pld, &rule, &rule_count);
+
+ for (current = 0; current < rule_count; current++)
+ {
+ switch (rule->type)
{
- void *output;
- int current;
-
- /* ok, do the parsing */
- output = allocator_alloc(payload_info->data_struct_length);
-
- for (current = 0; current < payload_info->encoding_rules_count; current++)
+ case U_INT_4:
{
- encoding_rule_t *rule = &(payload_info->ecoding_rules[current]);
- switch (rule->type)
+ u_int8_t *output_pos = output + rule->offset;
+ if (this->byte_pos + sizeof(u_int8_t) > this->input_roof)
{
- case U_INT_4:
- {
- u_int8_t *output_pos = output + rule->offset;
- if (context->byte_pos + sizeof(u_int8_t) > context->input_roof)
- {
- this->logger->log(this->logger, ERROR, "not enough input to parse U_INT_4");
- allocator_free(output);
- return PARSE_ERROR;
- }
- switch (context->bit_pos)
- {
- case 0:
- *output_pos = *(context->byte_pos) >> 4;
- context->bit_pos = 4;
- break;
- case 4:
- *output_pos = *(context->byte_pos) & 0x0F;
- context->bit_pos = 0;
- context->byte_pos++;
- break;
- default:
- this->logger->log(this->logger, ERROR, "found rule U_INT_4 on bitpos %d", context->bit_pos);
- allocator_free(output);
- return PARSE_ERROR;
- }
- break;
- }
- case U_INT_8:
- {
- u_int8_t *output_pos = output + rule->offset;
- if (context->byte_pos + sizeof(u_int8_t) > context->input_roof)
- {
- this->logger->log(this->logger, ERROR, "not enough input to parse U_INT_8");
- allocator_free(output);
- return PARSE_ERROR;
- }
- if (context->bit_pos)
- {
- this->logger->log(this->logger, ERROR, "found rule U_INT_8 on bitpos %d", context->bit_pos);
- allocator_free(output);
- return PARSE_ERROR;
- }
-
- *output_pos = *(context->byte_pos);
- context->byte_pos++;
- break;
- }
- case U_INT_16:
- {
- u_int16_t *output_pos = output + rule->offset;
- if (context->byte_pos + sizeof(u_int16_t) > context->input_roof)
- {
- this->logger->log(this->logger, ERROR, "not enough input to parse U_INT_16");
- allocator_free(output);
- return PARSE_ERROR;
- }
- if (context->bit_pos)
- {
- this->logger->log(this->logger, ERROR, "found rule U_INT_16 on bitpos %d", context->bit_pos);
- allocator_free(output);
- return PARSE_ERROR;
- }
- if ((int)context->byte_pos % 2)
- {
- this->logger->log(this->logger, ERROR, "found rule U_INT_16 on odd bytepos");
- allocator_free(output);
- return PARSE_ERROR;
- }
- *output_pos = ntohs(*((u_int16_t*)context->byte_pos));
- context->byte_pos += 2;
- break;
- }
- case U_INT_32:
- {
- u_int32_t *output_pos = output + rule->offset;
- if (context->byte_pos + sizeof(u_int32_t) > context->input_roof)
- {
- this->logger->log(this->logger, ERROR, "not enough input to parse U_INT_32");
- allocator_free(output);
- return PARSE_ERROR;
- }
- if (context->bit_pos)
- {
- this->logger->log(this->logger, ERROR, "found rule U_INT_32 on bitpos %d", context->bit_pos);
- allocator_free(output);
- return PARSE_ERROR;
- }
- if ((int)context->byte_pos % 4)
- {
- this->logger->log(this->logger, ERROR, "found rule U_INT_32 on unaligned bytepos");
- allocator_free(output);
- return PARSE_ERROR;
- }
- *output_pos = ntohl(*((u_int32_t*)context->byte_pos));
- context->byte_pos += 4;
- break;
- }
- case U_INT_64:
- {
- u_int32_t *output_pos = output + rule->offset;
- if (context->byte_pos + 2 * sizeof(u_int32_t) > context->input_roof)
- {
- this->logger->log(this->logger, ERROR, "not enough input to parse U_INT_64");
- allocator_free(output);
- return PARSE_ERROR;
- }
- if (context->bit_pos)
- {
- this->logger->log(this->logger, ERROR, "found rule U_INT_64 on bitpos %d", context->bit_pos);
- allocator_free(output);
- return PARSE_ERROR;
- }
- if ((int)context->byte_pos % 8)
- {
- this->logger->log(this->logger, ERROR, "found rule U_INT_64 on unaligned bytepos");
- allocator_free(output);
- return PARSE_ERROR;
- }
- /* assuming little endian host order */
- *(output_pos + 1) = ntohl(*((u_int32_t*)context->byte_pos));
- context->byte_pos += 4;
- *output_pos = ntohl(*((u_int32_t*)context->byte_pos));
- context->byte_pos += 4;
-
- break;
- }
- case RESERVED_BIT:
- {
- if (context->byte_pos > context->input_roof)
- {
- this->logger->log(this->logger, ERROR, "not enough input to parse RESERVED_BIT");
- allocator_free(output);
- return PARSE_ERROR;
- }
- context->bit_pos = (context->bit_pos + 1) % 8;
- if (context->bit_pos == 0)
- {
- context->byte_pos++;
- }
- break;
- }
- case RESERVED_BYTE:
- {
- if (context->byte_pos > context->input_roof)
- {
- this->logger->log(this->logger, ERROR, "not enough input to parse RESERVED_BYTE");
- allocator_free(output);
- return PARSE_ERROR;
- }
- if (context->bit_pos)
- {
- this->logger->log(this->logger, ERROR, "found rule RESERVED_BYTE on bitpos %d", context->bit_pos);
- allocator_free(output);
- return PARSE_ERROR;
- }
- context->byte_pos++;
+ this->logger->log(this->logger, ERROR, "not enough input to parse U_INT_4");
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
+ switch (this->bit_pos)
+ {
+ case 0:
+ *output_pos = *(this->byte_pos) >> 4;
+ this->bit_pos = 4;
break;
- }
- case FLAG:
- {
- bool *output_pos = output + rule->offset;
- u_int8_t mask;
- if (context->byte_pos > context->input_roof)
- {
- this->logger->log(this->logger, ERROR, "not enough input to parse FLAG");
- allocator_free(output);
- return PARSE_ERROR;
- }
- mask = 0x01 << (7 - context->bit_pos);
- *output_pos = *context->byte_pos & mask;
-
- if (*output_pos)
- {
- /* set to a "clean", comparable true */
- *output_pos = TRUE;
- }
- context->bit_pos = (context->bit_pos + 1) % 8;
- if (context->bit_pos == 0)
- {
- context->byte_pos++;
- }
+ case 4:
+ *output_pos = *(this->byte_pos) & 0x0F;
+ this->bit_pos = 0;
+ this->byte_pos++;
break;
- }
- case LENGTH:
- {
- u_int32_t *output_pos = output + rule->offset;
- if (context->byte_pos + sizeof(u_int32_t) > context->input_roof)
- {
- this->logger->log(this->logger, ERROR, "not enough input to parse LENGTH");
- allocator_free(output);
- return PARSE_ERROR;
- }
- if (context->bit_pos)
- {
- this->logger->log(this->logger, ERROR, "found rule LENGTH on bitpos %d", context->bit_pos);
- allocator_free(output);
- return PARSE_ERROR;
- }
- if ((int)context->byte_pos % 4)
- {
- this->logger->log(this->logger, ERROR, "found rule LENGTH on unaligned bytepos");
- allocator_free(output);
- return PARSE_ERROR;
- }
- *output_pos = ntohl(*((u_int32_t*)context->byte_pos));
- context->byte_pos += 4;
- break;
-
- }
- case SPI_SIZE:
- {
-
- }
default:
- {
- this->logger->log(this->logger, ERROR, "parser found unknown type");
- allocator_free(output);
+ this->logger->log(this->logger, ERROR, "found rule U_INT_4 on bitpos %d", this->bit_pos);
+ pld->destroy(pld);
return PARSE_ERROR;
- }
- }
+ }
+ break;
+ }
+ case U_INT_8:
+ {
+ u_int8_t *output_pos = output + rule->offset;
+ if (this->byte_pos + sizeof(u_int8_t) > this->input_roof)
+ {
+ this->logger->log(this->logger, ERROR, "not enough input to parse U_INT_8");
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
+ if (this->bit_pos)
+ {
+ this->logger->log(this->logger, ERROR, "found rule U_INT_8 on bitpos %d", this->bit_pos);
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
+
+ *output_pos = *(this->byte_pos);
+ this->byte_pos++;
+ break;
+ }
+ case U_INT_16:
+ {
+ u_int16_t *output_pos = output + rule->offset;
+ if (this->byte_pos + sizeof(u_int16_t) > this->input_roof)
+ {
+ this->logger->log(this->logger, ERROR, "not enough input to parse U_INT_16");
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
+ if (this->bit_pos)
+ {
+ this->logger->log(this->logger, ERROR, "found rule U_INT_16 on bitpos %d", this->bit_pos);
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
+ if ((int)this->byte_pos % 2)
+ {
+ this->logger->log(this->logger, ERROR, "found rule U_INT_16 on odd bytepos");
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
+ *output_pos = ntohs(*((u_int16_t*)this->byte_pos));
+ this->byte_pos += 2;
+ break;
+ }
+ case U_INT_32:
+ {
+ u_int32_t *output_pos = output + rule->offset;
+ if (this->byte_pos + sizeof(u_int32_t) > this->input_roof)
+ {
+ this->logger->log(this->logger, ERROR, "not enough input to parse U_INT_32");
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
+ if (this->bit_pos)
+ {
+ this->logger->log(this->logger, ERROR, "found rule U_INT_32 on bitpos %d", this->bit_pos);
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
+ if ((int)this->byte_pos % 4)
+ {
+ this->logger->log(this->logger, ERROR, "found rule U_INT_32 on unaligned bytepos");
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
+ *output_pos = ntohl(*((u_int32_t*)this->byte_pos));
+ this->byte_pos += 4;
+ break;
+ }
+ case U_INT_64:
+ {
+ u_int32_t *output_pos = output + rule->offset;
+ if (this->byte_pos + 2 * sizeof(u_int32_t) > this->input_roof)
+ {
+ this->logger->log(this->logger, ERROR, "not enough input to parse U_INT_64");
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
+ if (this->bit_pos)
+ {
+ this->logger->log(this->logger, ERROR, "found rule U_INT_64 on bitpos %d", this->bit_pos);
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
+ if ((int)this->byte_pos % 8)
+ {
+ this->logger->log(this->logger, ERROR, "found rule U_INT_64 on unaligned bytepos");
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
+ /* assuming little endian host order */
+ *(output_pos + 1) = ntohl(*((u_int32_t*)this->byte_pos));
+ this->byte_pos += 4;
+ *output_pos = ntohl(*((u_int32_t*)this->byte_pos));
+ this->byte_pos += 4;
+
+ break;
}
+ case RESERVED_BIT:
+ {
+ if (this->byte_pos > this->input_roof)
+ {
+ this->logger->log(this->logger, ERROR, "not enough input to parse RESERVED_BIT");
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
+ this->bit_pos = (this->bit_pos + 1) % 8;
+ if (this->bit_pos == 0)
+ {
+ this->byte_pos++;
+ }
+ break;
+ }
+ case RESERVED_BYTE:
+ {
+ if (this->byte_pos > this->input_roof)
+ {
+ this->logger->log(this->logger, ERROR, "not enough input to parse RESERVED_BYTE");
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
+ if (this->bit_pos)
+ {
+ this->logger->log(this->logger, ERROR, "found rule RESERVED_BYTE on bitpos %d", this->bit_pos);
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
+ this->byte_pos++;
+ break;
+ }
+ case FLAG:
+ {
+ bool *output_pos = output + rule->offset;
+ u_int8_t mask;
+ if (this->byte_pos > this->input_roof)
+ {
+ this->logger->log(this->logger, ERROR, "not enough input to parse FLAG");
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
+ mask = 0x01 << (7 - this->bit_pos);
+ *output_pos = *this->byte_pos & mask;
- *data_struct = output;
- return SUCCESS;
+ if (*output_pos)
+ {
+ /* set to a "clean", comparable true */
+ *output_pos = TRUE;
+ }
+ this->bit_pos = (this->bit_pos + 1) % 8;
+ if (this->bit_pos == 0)
+ {
+ this->byte_pos++;
+ }
+ break;
+ }
+ case LENGTH:
+ {
+ u_int32_t *output_pos = output + rule->offset;
+ if (this->byte_pos + sizeof(u_int32_t) > this->input_roof)
+ {
+ this->logger->log(this->logger, ERROR, "not enough input to parse LENGTH");
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
+ if (this->bit_pos)
+ {
+ this->logger->log(this->logger, ERROR, "found rule LENGTH on bitpos %d", this->bit_pos);
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
+ if ((int)this->byte_pos % 4)
+ {
+ this->logger->log(this->logger, ERROR, "found rule LENGTH on unaligned bytepos");
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
+ *output_pos = ntohl(*((u_int32_t*)this->byte_pos));
+ this->byte_pos += 4;
+ break;
+
+ }
+ case SPI_SIZE:
+ {
+
+ }
+ default:
+ {
+ this->logger->log(this->logger, ERROR, "parser found unknown type");
+ pld->destroy(pld);
+ return PARSE_ERROR;
+ }
}
- payload_info++;
+ /* process next rulue */
+ rule++;
}
- this->logger->log(this->logger, ERROR, "Payload not supported");
- return NOT_SUPPORTED;
+ *payload = pld;
+ return SUCCESS;
}
/**
/*
* see header file
*/
-parser_t *parser_create(payload_info_t **payload_infos)
+parser_t *parser_create(chunk_t data)
{
private_parser_t *this = allocator_alloc_thing(private_parser_t);
return NULL;
}
- this->logger = global_logger_manager->create_logger(global_logger_manager,PARSER, NULL);
+ this->logger = global_logger_manager->create_logger(global_logger_manager, PARSER, NULL);
if (this->logger == NULL)
{
allocator_free(this);
return NULL;
}
- this->public.create_context = (parser_context_t*(*)(parser_t*,chunk_t)) create_context;
- this->public.parse_payload = (status_t(*)(parser_t*,payload_type_t,void**,parser_context_t*)) parse_payload;
+
+ this->public.parse_payload = (status_t(*)(parser_t*,payload_type_t,payload_t**)) parse_payload;
this->public.destroy = (status_t(*)(parser_t*)) destroy;
- this->payload_infos = payload_infos;
+ this->input = data.ptr;
+ this->byte_pos = data.ptr;
+ this->bit_pos = 0;
+ this->input_roof = data.ptr + data.len;
return (parser_t*)this;
}
+
#include "../payloads/encodings.h"
#include "../payloads/ike_header.h"
-extern payload_info_t *payload_infos[];
extern logger_manager_t *global_logger_manager;
void test_parser_with_header_payload(tester_t *tester)
{
parser_t *parser;
- parser_context_t *parser_context;
- ike_header_t *header_data;
+ ike_header_t *ike_header;
status_t status;
- chunk_t test_chunk;
+ chunk_t header_chunk;
- logger_t *logger;
-
- logger = global_logger_manager->create_logger(global_logger_manager,TESTER, "header payload");
-
- u_int8_t test_bytes[] = {
+ u_int8_t header_bytes[] = {
0x00,0x00,0x00,0x00,
0x00,0x00,0x00,0x01,
0x00,0x00,0x00,0x00,
0x00,0x00,0x00,0x07,
0x00,0x00,0x00,0x08,
};
- test_chunk.ptr = test_bytes;
- test_chunk.len = sizeof(test_bytes);
+ header_chunk.ptr = header_bytes;
+ header_chunk.len = sizeof(header_bytes);
- parser = parser_create(payload_infos);
+ parser = parser_create(header_chunk);
tester->assert_true(tester,(parser != NULL), "parser create check");
- parser_context = parser->create_context(parser, test_chunk);
- tester->assert_true(tester,(parser_context != NULL), "parser_context create check");
-
- status = parser->parse_payload(parser, HEADER, (void**)&header_data, parser_context);
+ status = parser->parse_payload(parser, HEADER, (payload_t**)&ike_header);
tester->assert_true(tester,(status == SUCCESS),"parse_payload call check");
- tester->assert_true(tester,(header_data->initiator_spi == 1),"parsed initiator_spi value");
- tester->assert_true(tester,(header_data->responder_spi == 2),"parsed responder_spi value");
- tester->assert_true(tester,(header_data->next_payload == 3),"parsed next_payload value");
- tester->assert_true(tester,(header_data->maj_version == 4),"parsed maj_version value");
- tester->assert_true(tester,(header_data->min_version == 5),"parsed min_version value");
- tester->assert_true(tester,(header_data->exchange_type == 6),"parsed exchange_type value");
- tester->assert_true(tester,(header_data->flags.initiator == TRUE),"parsed flags.initiator value");
- tester->assert_true(tester,(header_data->flags.version == FALSE),"parsed flags.version value");
- tester->assert_true(tester,(header_data->flags.response == TRUE),"parsed flags.response value");
- tester->assert_true(tester,(header_data->message_id == 7),"parsed message_id value");
- tester->assert_true(tester,(header_data->length == 8),"parsed length value");
+ tester->assert_true(tester,(ike_header->initiator_spi == 1),"parsed initiator_spi value");
+ tester->assert_true(tester,(ike_header->responder_spi == 2),"parsed responder_spi value");
+ tester->assert_true(tester,(ike_header->next_payload == 3),"parsed next_payload value");
+ tester->assert_true(tester,(ike_header->maj_version == 4),"parsed maj_version value");
+ tester->assert_true(tester,(ike_header->min_version == 5),"parsed min_version value");
+ tester->assert_true(tester,(ike_header->exchange_type == 6),"parsed exchange_type value");
+ tester->assert_true(tester,(ike_header->flags.initiator == TRUE),"parsed flags.initiator value");
+ tester->assert_true(tester,(ike_header->flags.version == FALSE),"parsed flags.version value");
+ tester->assert_true(tester,(ike_header->flags.response == TRUE),"parsed flags.response value");
+ tester->assert_true(tester,(ike_header->message_id == 7),"parsed message_id value");
+ tester->assert_true(tester,(ike_header->length == 8),"parsed length value");
-
- parser_context->destroy(parser_context);
tester->assert_true(tester,(parser->destroy(parser) == SUCCESS), "parser destroy call check");
- logger->log_bytes(logger, RAW, "Header", (void*)header_data, sizeof(ike_header_t));
-
- allocator_free(header_data);
+ ike_header->destroy(ike_header);
}