Add basic sum parsing
This commit is contained in:
79
lib/reader.c
79
lib/reader.c
@@ -3,19 +3,76 @@
|
|||||||
#include <ctype.h>
|
#include <ctype.h>
|
||||||
#include <stddef.h>
|
#include <stddef.h>
|
||||||
|
|
||||||
|
static int parse_number(
|
||||||
|
memory_pool_t *pool, const char *input, int remaining, expression_t *out)
|
||||||
|
{
|
||||||
|
(void)pool;
|
||||||
|
|
||||||
|
int value = 0, used = 0;
|
||||||
|
while (remaining > 0 && isdigit(*input)) {
|
||||||
|
value *= 10;
|
||||||
|
value += *input - '0';
|
||||||
|
++used;
|
||||||
|
--remaining;
|
||||||
|
++input;
|
||||||
|
}
|
||||||
|
|
||||||
|
out->is_number = true;
|
||||||
|
out->number = value;
|
||||||
|
return used;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int parse_application(
|
||||||
|
memory_pool_t *pool, const char *input, int remaining, expression_t *out)
|
||||||
|
{
|
||||||
|
int result, used = 0;
|
||||||
|
|
||||||
|
expression_t *operand0 = allocate_expression(pool);
|
||||||
|
result = parse_number(pool, input, remaining, operand0);
|
||||||
|
if (0 == result)
|
||||||
|
return 0;
|
||||||
|
used += result;
|
||||||
|
remaining -= result;
|
||||||
|
input += result;
|
||||||
|
|
||||||
|
if (remaining <= 0 || *input != '+')
|
||||||
|
return 0;
|
||||||
|
++used;
|
||||||
|
--remaining;
|
||||||
|
++input;
|
||||||
|
|
||||||
|
expression_t *operand1 = allocate_expression(pool);
|
||||||
|
result = parse_number(pool, input, remaining, operand1);
|
||||||
|
if (0 == result)
|
||||||
|
return 0;
|
||||||
|
used += result;
|
||||||
|
remaining -= result;
|
||||||
|
input += result;
|
||||||
|
|
||||||
|
out->is_number = false;
|
||||||
|
out->application.operator = OPERATOR_ADD;
|
||||||
|
out->application.operands[0] = operand0;
|
||||||
|
out->application.operands[1] = operand1;
|
||||||
|
|
||||||
|
return used;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int parse_expression(
|
||||||
|
memory_pool_t *pool, const char *input, int remaining, expression_t *out)
|
||||||
|
{
|
||||||
|
int result = parse_application(pool, input, remaining, out);
|
||||||
|
if (0 != result)
|
||||||
|
return result;
|
||||||
|
else
|
||||||
|
return parse_number(pool, input, remaining, out);
|
||||||
|
}
|
||||||
|
|
||||||
const expression_t *
|
const expression_t *
|
||||||
read_expression(memory_pool_t *pool, const char *input, int len)
|
read_expression(memory_pool_t *pool, const char *input, int len)
|
||||||
{
|
{
|
||||||
expression_t *result = allocate_expression(pool);
|
expression_t *expression = allocate_expression(pool);
|
||||||
if (NULL == result)
|
if (NULL == expression)
|
||||||
return NULL;
|
return NULL;
|
||||||
result->is_number = true;
|
parse_expression(pool, input, len, expression);
|
||||||
result->number = 0;
|
return expression;
|
||||||
while (isdigit(*input)) {
|
|
||||||
result->number *= 10;
|
|
||||||
result->number += *input - '0';
|
|
||||||
++input;
|
|
||||||
++len;
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,10 +21,30 @@ static void input_4321_is_read_as_number_with_expected_value(void)
|
|||||||
ASSERT_EQUAL(4321, result->number);
|
ASSERT_EQUAL(4321, result->number);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void input_1234_with_len_3_is_read_as_123(void)
|
||||||
|
{
|
||||||
|
init_memory_pool(&pool);
|
||||||
|
const expression_t *result = read_expression(&pool, "1234", 3);
|
||||||
|
ASSERT_NOT_NULL(result);
|
||||||
|
ASSERT_TRUE(result->is_number);
|
||||||
|
ASSERT_EQUAL(123, result->number);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void input_1_plus_2_with_no_spaces_is_read_as_an_add_application(void)
|
||||||
|
{
|
||||||
|
init_memory_pool(&pool);
|
||||||
|
const expression_t *result = read_expression(&pool, "1+2", 3);
|
||||||
|
ASSERT_NOT_NULL(result);
|
||||||
|
ASSERT_FALSE(result->is_number);
|
||||||
|
ASSERT_EQUAL(OPERATOR_ADD, result->application.operator);
|
||||||
|
}
|
||||||
|
|
||||||
int main(void)
|
int main(void)
|
||||||
{
|
{
|
||||||
TESTING_BEGIN();
|
TESTING_BEGIN();
|
||||||
RUN_TEST(input_1234_is_read_as_number_with_expected_value);
|
RUN_TEST(input_1234_is_read_as_number_with_expected_value);
|
||||||
RUN_TEST(input_4321_is_read_as_number_with_expected_value);
|
RUN_TEST(input_4321_is_read_as_number_with_expected_value);
|
||||||
|
RUN_TEST(input_1234_with_len_3_is_read_as_123);
|
||||||
|
RUN_TEST(input_1_plus_2_with_no_spaces_is_read_as_an_add_application);
|
||||||
TESTING_END();
|
TESTING_END();
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user