2016-10-16 20:42:55 -07:00
|
|
|
#ifndef RUNTIME_TREE_PATH_H_
|
|
|
|
|
#define RUNTIME_TREE_PATH_H_
|
|
|
|
|
|
|
|
|
|
#include "runtime/tree.h"
|
2017-07-17 17:07:56 -07:00
|
|
|
#include "runtime/language.h"
|
2016-10-16 21:10:25 -07:00
|
|
|
#include "runtime/error_costs.h"
|
2016-10-16 20:42:55 -07:00
|
|
|
|
|
|
|
|
typedef Array(TSRange) RangeArray;
|
|
|
|
|
|
2016-11-04 09:18:38 -07:00
|
|
|
static void range_array_add(RangeArray *results, TSPoint start, TSPoint end) {
|
2016-10-16 20:42:55 -07:00
|
|
|
if (results->size > 0) {
|
|
|
|
|
TSRange *last_range = array_back(results);
|
2016-11-09 20:59:05 -08:00
|
|
|
if (point_lte(start, last_range->end)) {
|
2016-10-16 20:42:55 -07:00
|
|
|
last_range->end = end;
|
2016-11-04 09:18:38 -07:00
|
|
|
return;
|
2016-10-16 20:42:55 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-09 20:59:05 -08:00
|
|
|
if (point_lt(start, end)) {
|
2016-10-16 20:42:55 -07:00
|
|
|
TSRange range = { start, end };
|
2016-11-04 09:18:38 -07:00
|
|
|
array_push(results, range);
|
2016-10-16 20:42:55 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-01-24 12:48:47 -08:00
|
|
|
static bool tree_path_descend(TreePath *path, Length position) {
|
2016-11-14 12:15:24 -08:00
|
|
|
uint32_t original_size = path->size;
|
2017-01-24 12:48:47 -08:00
|
|
|
|
2016-10-16 20:42:55 -07:00
|
|
|
bool did_descend;
|
|
|
|
|
do {
|
|
|
|
|
did_descend = false;
|
|
|
|
|
TreePathEntry entry = *array_back(path);
|
2017-01-24 12:48:47 -08:00
|
|
|
Length child_left = entry.position;
|
2016-11-14 12:15:24 -08:00
|
|
|
for (uint32_t i = 0; i < entry.tree->child_count; i++) {
|
2016-11-09 20:59:05 -08:00
|
|
|
Tree *child = entry.tree->children[i];
|
2017-01-24 12:48:47 -08:00
|
|
|
Length child_right = length_add(child_left, ts_tree_total_size(child));
|
|
|
|
|
if (position.bytes < child_right.bytes) {
|
|
|
|
|
TreePathEntry child_entry = { child, child_left, i };
|
|
|
|
|
if (child->visible || child->child_count == 0) {
|
2016-10-16 20:42:55 -07:00
|
|
|
array_push(path, child_entry);
|
|
|
|
|
return true;
|
2017-01-24 12:48:47 -08:00
|
|
|
} else if (child->visible_child_count > 0) {
|
2016-10-16 20:42:55 -07:00
|
|
|
array_push(path, child_entry);
|
|
|
|
|
did_descend = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-01-24 12:48:47 -08:00
|
|
|
child_left = child_right;
|
2016-10-16 20:42:55 -07:00
|
|
|
}
|
|
|
|
|
} while (did_descend);
|
2017-01-24 12:48:47 -08:00
|
|
|
|
2016-11-05 21:26:01 -07:00
|
|
|
path->size = original_size;
|
2016-10-16 20:42:55 -07:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-14 12:15:24 -08:00
|
|
|
static uint32_t tree_path_advance(TreePath *path) {
|
|
|
|
|
uint32_t ascend_count = 0;
|
2017-01-24 12:48:47 -08:00
|
|
|
|
2016-10-16 20:42:55 -07:00
|
|
|
while (path->size > 0) {
|
|
|
|
|
TreePathEntry entry = array_pop(path);
|
2017-01-24 12:48:47 -08:00
|
|
|
if (path->size == 0) break;
|
2016-10-16 20:42:55 -07:00
|
|
|
TreePathEntry parent_entry = *array_back(path);
|
|
|
|
|
if (parent_entry.tree->visible) ascend_count++;
|
2017-01-24 12:48:47 -08:00
|
|
|
|
|
|
|
|
Length position = length_add(entry.position, ts_tree_total_size(entry.tree));
|
2016-11-14 12:15:24 -08:00
|
|
|
for (uint32_t i = entry.child_index + 1; i < parent_entry.tree->child_count; i++) {
|
2016-11-09 20:59:05 -08:00
|
|
|
Tree *next_child = parent_entry.tree->children[i];
|
2017-01-24 12:48:47 -08:00
|
|
|
if (next_child->visible ||
|
|
|
|
|
next_child->child_count == 0 ||
|
|
|
|
|
next_child->visible_child_count > 0) {
|
2016-10-16 20:42:55 -07:00
|
|
|
if (parent_entry.tree->visible) ascend_count--;
|
|
|
|
|
array_push(path, ((TreePathEntry){
|
|
|
|
|
.tree = next_child,
|
|
|
|
|
.child_index = i,
|
|
|
|
|
.position = position,
|
|
|
|
|
}));
|
2017-01-24 12:48:47 -08:00
|
|
|
if (!next_child->visible) {
|
|
|
|
|
tree_path_descend(path, length_zero());
|
|
|
|
|
}
|
2016-10-16 20:42:55 -07:00
|
|
|
return ascend_count;
|
|
|
|
|
}
|
2016-11-09 20:59:05 -08:00
|
|
|
position = length_add(position, ts_tree_total_size(next_child));
|
2016-10-16 20:42:55 -07:00
|
|
|
}
|
|
|
|
|
}
|
2017-01-24 12:48:47 -08:00
|
|
|
|
2016-10-16 20:42:55 -07:00
|
|
|
return ascend_count;
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-14 12:15:24 -08:00
|
|
|
static void tree_path_ascend(TreePath *path, uint32_t count) {
|
|
|
|
|
for (uint32_t i = 0; i < count; i++) {
|
2016-10-16 20:42:55 -07:00
|
|
|
do {
|
2016-11-05 14:39:25 -07:00
|
|
|
path->size--;
|
2016-10-16 20:42:55 -07:00
|
|
|
} while (path->size > 0 && !array_back(path)->tree->visible);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-09 20:59:05 -08:00
|
|
|
static void tree_path_init(TreePath *path, Tree *tree) {
|
2016-10-16 20:42:55 -07:00
|
|
|
array_clear(path);
|
2016-11-09 20:19:02 -08:00
|
|
|
array_push(path, ((TreePathEntry){
|
|
|
|
|
.tree = tree,
|
|
|
|
|
.position = { 0, 0, { 0, 0 } },
|
|
|
|
|
.child_index = 0,
|
|
|
|
|
}));
|
2017-01-24 12:48:47 -08:00
|
|
|
if (!tree->visible) {
|
|
|
|
|
tree_path_descend(path, length_zero());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Tree *tree_path_visible_tree(TreePath *self) {
|
|
|
|
|
for (uint32_t i = self->size - 1; i + 1 > 0; i--) {
|
|
|
|
|
Tree *tree = self->contents[i].tree;
|
|
|
|
|
if (tree->visible) return tree;
|
|
|
|
|
}
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Length tree_path_start_position(TreePath *self) {
|
|
|
|
|
TreePathEntry entry = *array_back(self);
|
|
|
|
|
return length_add(entry.position, entry.tree->padding);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Length tree_path_end_position(TreePath *self) {
|
|
|
|
|
TreePathEntry entry = *array_back(self);
|
|
|
|
|
return length_add(length_add(entry.position, entry.tree->padding), entry.tree->size);
|
2016-10-16 20:42:55 -07:00
|
|
|
}
|
|
|
|
|
|
2017-07-17 17:07:56 -07:00
|
|
|
typedef enum {
|
|
|
|
|
TreePathNotEq,
|
|
|
|
|
TreePathCanEq,
|
|
|
|
|
TreePathMustEq,
|
|
|
|
|
} TreePathComparison;
|
|
|
|
|
|
|
|
|
|
TreePathComparison tree_path_compare(const TreePath *old_path,
|
|
|
|
|
const TreePath *new_path,
|
|
|
|
|
const TSLanguage *language) {
|
|
|
|
|
Tree *old_tree = NULL;
|
|
|
|
|
TSSymbol old_rename_symbol = 0;
|
|
|
|
|
Length old_start = length_zero();
|
|
|
|
|
for (uint32_t i = old_path->size - 1; i + 1 > 0; i--) {
|
|
|
|
|
old_tree = old_path->contents[i].tree;
|
|
|
|
|
if (old_tree->visible) {
|
|
|
|
|
old_start = old_path->contents[i].position;
|
|
|
|
|
if (i > 0) {
|
|
|
|
|
const TSSymbol *rename_sequence = ts_language_rename_sequence(
|
|
|
|
|
language,
|
|
|
|
|
old_path->contents[i - 1].tree->rename_sequence_id
|
|
|
|
|
);
|
|
|
|
|
if (rename_sequence) {
|
|
|
|
|
old_rename_symbol = rename_sequence[old_path->contents[i].child_index];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Tree *new_tree = NULL;
|
|
|
|
|
TSSymbol new_rename_symbol = 0;
|
|
|
|
|
Length new_start = length_zero();
|
|
|
|
|
for (uint32_t i = new_path->size - 1; i + 1 > 0; i--) {
|
|
|
|
|
new_tree = new_path->contents[i].tree;
|
|
|
|
|
if (new_tree->visible) {
|
|
|
|
|
new_start = old_path->contents[i].position;
|
|
|
|
|
if (i > 0) {
|
|
|
|
|
const TSSymbol *rename_sequence = ts_language_rename_sequence(
|
|
|
|
|
language,
|
|
|
|
|
new_path->contents[i - 1].tree->rename_sequence_id
|
|
|
|
|
);
|
|
|
|
|
if (rename_sequence) {
|
|
|
|
|
new_rename_symbol = rename_sequence[new_path->contents[i].child_index];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (old_rename_symbol == new_rename_symbol) {
|
|
|
|
|
if (old_start.bytes == new_start.bytes) {
|
|
|
|
|
if (!old_tree->has_changes &&
|
|
|
|
|
old_tree->symbol == new_tree->symbol &&
|
|
|
|
|
old_tree->symbol != ts_builtin_sym_error &&
|
|
|
|
|
old_tree->size.bytes == new_tree->size.bytes &&
|
|
|
|
|
old_tree->parse_state != TS_TREE_STATE_NONE &&
|
|
|
|
|
new_tree->parse_state != TS_TREE_STATE_NONE &&
|
|
|
|
|
(old_tree->parse_state == ERROR_STATE) == (new_tree->parse_state == ERROR_STATE)) {
|
|
|
|
|
return TreePathMustEq;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (old_tree->symbol == new_tree->symbol) {
|
|
|
|
|
return TreePathCanEq;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return TreePathNotEq;
|
2016-10-16 20:42:55 -07:00
|
|
|
}
|
|
|
|
|
|
2016-11-04 09:18:38 -07:00
|
|
|
static void tree_path_get_changes(TreePath *old_path, TreePath *new_path,
|
2017-07-17 17:07:56 -07:00
|
|
|
TSRange **ranges, uint32_t *range_count,
|
|
|
|
|
const TSLanguage *language) {
|
2017-01-24 12:48:47 -08:00
|
|
|
Length position = length_zero();
|
2016-10-16 20:42:55 -07:00
|
|
|
RangeArray results = array_new();
|
|
|
|
|
|
|
|
|
|
while (old_path->size && new_path->size) {
|
|
|
|
|
bool is_changed = false;
|
2017-01-24 12:48:47 -08:00
|
|
|
Length next_position = position;
|
|
|
|
|
|
|
|
|
|
Length old_start = tree_path_start_position(old_path);
|
|
|
|
|
Length new_start = tree_path_start_position(new_path);
|
|
|
|
|
Length old_end = tree_path_end_position(old_path);
|
|
|
|
|
Length new_end = tree_path_end_position(new_path);
|
2016-10-16 20:42:55 -07:00
|
|
|
|
2016-11-09 20:59:05 -08:00
|
|
|
// #define NAME(t) (ts_language_symbol_name(language, ((Tree *)(t))->symbol))
|
2017-01-24 12:48:47 -08:00
|
|
|
// printf("At [%-2u, %-2u] Compare (%-20s\t [%-2u, %-2u] - [%u, %u])\tvs\t(%-20s\t [%u, %u] - [%u, %u])\n",
|
|
|
|
|
// position.extent.row, position.extent.column,
|
|
|
|
|
// NAME(old_tree), old_start.extent.row, old_start.extent.column, old_end.extent.row, old_end.extent.column,
|
|
|
|
|
// NAME(new_tree), new_start.extent.row, new_start.extent.column, new_end.extent.row, new_end.extent.column);
|
|
|
|
|
|
|
|
|
|
if (position.bytes < old_start.bytes) {
|
|
|
|
|
if (position.bytes < new_start.bytes) {
|
|
|
|
|
next_position = length_min(old_start, new_start);
|
2016-10-16 20:42:55 -07:00
|
|
|
} else {
|
|
|
|
|
is_changed = true;
|
2017-01-24 12:48:47 -08:00
|
|
|
next_position = old_start;
|
2016-10-16 20:42:55 -07:00
|
|
|
}
|
2017-01-24 12:48:47 -08:00
|
|
|
} else if (position.bytes < new_start.bytes) {
|
2016-10-16 20:42:55 -07:00
|
|
|
is_changed = true;
|
2017-01-24 12:48:47 -08:00
|
|
|
next_position = new_start;
|
2017-07-17 17:07:56 -07:00
|
|
|
} else {
|
|
|
|
|
switch (tree_path_compare(old_path, new_path, language)) {
|
|
|
|
|
case TreePathMustEq:
|
|
|
|
|
next_position = old_end;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case TreePathCanEq:
|
|
|
|
|
if (tree_path_descend(old_path, position)) {
|
|
|
|
|
if (!tree_path_descend(new_path, position)) {
|
|
|
|
|
tree_path_ascend(old_path, 1);
|
|
|
|
|
is_changed = true;
|
|
|
|
|
next_position = new_end;
|
|
|
|
|
}
|
|
|
|
|
} else if (tree_path_descend(new_path, position)) {
|
|
|
|
|
tree_path_ascend(new_path, 1);
|
|
|
|
|
is_changed = true;
|
|
|
|
|
next_position = old_end;
|
|
|
|
|
} else {
|
|
|
|
|
next_position = length_min(old_end, new_end);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case TreePathNotEq:
|
2016-10-16 20:42:55 -07:00
|
|
|
is_changed = true;
|
2017-07-17 17:07:56 -07:00
|
|
|
next_position = length_min(old_end, new_end);
|
|
|
|
|
break;
|
2016-10-16 20:42:55 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-01-24 12:48:47 -08:00
|
|
|
bool at_old_end = old_end.bytes <= next_position.bytes;
|
|
|
|
|
bool at_new_end = new_end.bytes <= next_position.bytes;
|
2016-10-16 20:42:55 -07:00
|
|
|
|
|
|
|
|
if (at_new_end && at_old_end) {
|
2016-11-14 12:15:24 -08:00
|
|
|
uint32_t old_ascend_count = tree_path_advance(old_path);
|
|
|
|
|
uint32_t new_ascend_count = tree_path_advance(new_path);
|
2016-10-16 20:42:55 -07:00
|
|
|
if (old_ascend_count > new_ascend_count) {
|
|
|
|
|
tree_path_ascend(new_path, old_ascend_count - new_ascend_count);
|
|
|
|
|
} else if (new_ascend_count > old_ascend_count) {
|
|
|
|
|
tree_path_ascend(old_path, new_ascend_count - old_ascend_count);
|
|
|
|
|
}
|
|
|
|
|
} else if (at_new_end) {
|
2016-11-14 12:15:24 -08:00
|
|
|
uint32_t ascend_count = tree_path_advance(new_path);
|
2016-10-16 20:42:55 -07:00
|
|
|
tree_path_ascend(old_path, ascend_count);
|
|
|
|
|
} else if (at_old_end) {
|
2016-11-14 12:15:24 -08:00
|
|
|
uint32_t ascend_count = tree_path_advance(old_path);
|
2016-10-16 20:42:55 -07:00
|
|
|
tree_path_ascend(new_path, ascend_count);
|
|
|
|
|
}
|
|
|
|
|
|
2017-01-24 12:48:47 -08:00
|
|
|
if (is_changed) range_array_add(&results, position.extent, next_position.extent);
|
2016-10-16 20:42:55 -07:00
|
|
|
position = next_position;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
*ranges = results.contents;
|
|
|
|
|
*range_count = results.size;
|
|
|
|
|
}
|
|
|
|
|
|
2016-10-16 21:21:21 -07:00
|
|
|
#endif // RUNTIME_TREE_PATH_H_
|