aboutsummaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
authorMattias Andrée <maandree@operamail.com>2014-12-04 00:14:19 +0100
committerMattias Andrée <maandree@operamail.com>2014-12-04 00:14:22 +0100
commitd21c9495ad4dad01b428c9a9a66fc10f5c190b3a (patch)
tree67e9c41266f517f69d56785de3d16f75f75c94a8 /src
parentm (diff)
downloadmds-d21c9495ad4dad01b428c9a9a66fc10f5c190b3a.tar.gz
mds-d21c9495ad4dad01b428c9a9a66fc10f5c190b3a.tar.bz2
mds-d21c9495ad4dad01b428c9a9a66fc10f5c190b3a.tar.xz
mds-kbdc: compile have_range and if
Signed-off-by: Mattias Andrée <maandree@operamail.com>
Diffstat (limited to 'src')
-rw-r--r--src/mds-kbdc/compile-layout.c113
1 files changed, 105 insertions, 8 deletions
diff --git a/src/mds-kbdc/compile-layout.c b/src/mds-kbdc/compile-layout.c
index 88ef2fa..a82b52a 100644
--- a/src/mds-kbdc/compile-layout.c
+++ b/src/mds-kbdc/compile-layout.c
@@ -22,10 +22,16 @@
#include <stdlib.h>
#include <errno.h>
+#include <string.h>
/**
+ * This process's value for `mds_kbdc_tree_t.processed`
+ */
+#define PROCESS_LEVEL 6
+
+/**
* Tree type constant shortener
*/
#define C(TYPE) MDS_KBDC_TREE_TYPE_##TYPE
@@ -171,6 +177,7 @@ static int compile_country(mds_kbdc_tree_information_country_t* restrict tree)
fail_if ((data = parse_string((mds_kbdc_tree_t*)tree, tree->data, lineoff), data == NULL));
fail_if ((code = string_encode(data), code == NULL));
+ free(data);
result->countries[result->countries_ptr++] = code;
return 0;
@@ -218,7 +225,7 @@ static int compile_have_chars(mds_kbdc_tree_assumption_have_chars_t* restrict tr
size_t lineoff = tree->loc_end;
char* restrict code = result->source_code->real_lines[tree->loc_line];
char32_t* restrict data = NULL;
- char32_t** old32 = NULL;
+ char32_t** old = NULL;
char32_t* restrict character;
size_t n;
int saved_errno;
@@ -227,12 +234,12 @@ static int compile_have_chars(mds_kbdc_tree_assumption_have_chars_t* restrict tr
lineoff++;
fail_if ((data = parse_string((mds_kbdc_tree_t*)tree, tree->chars, lineoff), data == NULL));
- for (n = 0; data[n] != -1; n++);
+ for (n = 0; data[n] >= 0; n++);
if (result->assumed_strings_ptr + n >= result->assumed_strings_size)
{
result->assumed_strings_size += n;
- fail_if (xxrealloc(old32, result->assumed_strings, result->assumed_strings_size, char*));
+ fail_if (xxrealloc(old, result->assumed_strings, result->assumed_strings_size, char*));
}
while (n--)
@@ -243,10 +250,11 @@ static int compile_have_chars(mds_kbdc_tree_assumption_have_chars_t* restrict tr
result->assumed_strings[result->assumed_strings_ptr++] = character;
}
+ free(data);
return 0;
FAIL_BEGIN;
free(data);
- free(old32);
+ free(old);
FAIL_END;
}
@@ -259,8 +267,73 @@ static int compile_have_chars(mds_kbdc_tree_assumption_have_chars_t* restrict tr
*/
static int compile_have_range(mds_kbdc_tree_assumption_have_range_t* restrict tree)
{
- (void) tree;
- return 0; /* TODO */
+ size_t lineoff_first = tree->loc_end;
+ size_t lineoff_last;
+ char* restrict code = result->source_code->real_lines[tree->loc_line];
+ char32_t* restrict first = NULL;
+ char32_t* restrict last = NULL;
+ char32_t** old = NULL;
+ char32_t* restrict character;
+ size_t n;
+ int saved_errno;
+
+ while (code[lineoff_first] == ' ')
+ lineoff_first++;
+ fail_if ((first = parse_string((mds_kbdc_tree_t*)tree, tree->first, lineoff_first), first == NULL));
+
+ lineoff_last = lineoff_first + strlen(tree->first);
+
+ while (code[lineoff_last] == ' ')
+ lineoff_last++;
+ fail_if ((last = parse_string((mds_kbdc_tree_t*)tree, tree->last, lineoff_last), last == NULL));
+
+ if (tree->processed == PROCESS_LEVEL)
+ goto done;
+
+ if ((first[0] < 0) || (first[1] >= 0))
+ {
+ NEW_ERROR(tree, includes_ptr, ERROR, "iteration boundary must be a single character string");
+ error->start = lineoff_first, lineoff_first = 0;
+ error->end = error->start + strlen(tree->first);
+ }
+ if ((last[0] < 0) || (last[1] >= 0))
+ {
+ NEW_ERROR(tree, includes_ptr, ERROR, "iteration boundary must be a single character string");
+ error->start = lineoff_last, lineoff_last = 0;
+ error->end = error->start + strlen(tree->last);
+ }
+
+ if ((lineoff_first == 0) || (lineoff_last == 0))
+ goto done;
+
+ if (*first < *last)
+ *first ^= *last, *last ^= *first, *first ^= *last;
+
+ n = (size_t)(*last - *first) + 1;
+
+ if (result->assumed_strings_ptr + n >= result->assumed_strings_size)
+ {
+ result->assumed_strings_size += n;
+ fail_if (xxrealloc(old, result->assumed_strings, result->assumed_strings_size, char*));
+ }
+
+ while (*first != *last)
+ {
+ fail_if (xmalloc(character, 2, char32_t));
+ character[0] = (*first)++;
+ character[1] = -1;
+ result->assumed_strings[result->assumed_strings_ptr++] = character;
+ }
+
+ done:
+ free(first);
+ free(last);
+ return 0;
+ FAIL_BEGIN;
+ free(first);
+ free(last);
+ free(old);
+ FAIL_END;
}
@@ -311,8 +384,27 @@ static int compile_for(mds_kbdc_tree_for_t* restrict tree)
*/
static int compile_if(mds_kbdc_tree_if_t* restrict tree)
{
- (void) tree;
- return 0; /* TODO */
+ size_t lineoff = tree->loc_end;
+ char* restrict code = result->source_code->real_lines[tree->loc_line];
+ char32_t* restrict data = NULL;
+ int ok, saved_errno;
+ size_t i;
+
+ while (code[lineoff] == ' ')
+ lineoff++;
+
+ fail_if ((data = parse_string((mds_kbdc_tree_t*)tree, tree->condition, lineoff), data == NULL));
+ fail_if (tree->processed == PROCESS_LEVEL);
+
+ for (ok = 1, i = 0; data[i] >= 0; i++)
+ ok &= !!(data[i]);
+
+ free(data), data = NULL;
+
+ return compile_subtree(ok ? tree->inner : tree->otherwise);
+ FAIL_BEGIN;
+ free(data);
+ FAIL_END;
}
@@ -371,6 +463,9 @@ static int compile_subtree(mds_kbdc_tree_t* restrict tree)
if (tree == NULL)
return 0;
+ if (tree->processed == PROCESS_LEVEL)
+ goto next;
+
switch (tree->type)
{
case C(INFORMATION):
@@ -400,6 +495,7 @@ static int compile_subtree(mds_kbdc_tree_t* restrict tree)
break;
}
+ next:
if (break_level)
return 0;
@@ -431,4 +527,5 @@ int compile_layout(mds_kbdc_parsed_t* restrict result_)
#undef FAIL_BEGIN
#undef NEW_ERROR
#undef C
+#undef PROCESS_LEVEL