diff options
Diffstat (limited to 'src/tree_sitter/tree.c')
| -rw-r--r-- | src/tree_sitter/tree.c | 149 | 
1 files changed, 149 insertions, 0 deletions
| diff --git a/src/tree_sitter/tree.c b/src/tree_sitter/tree.c new file mode 100644 index 0000000000..04cb1d242f --- /dev/null +++ b/src/tree_sitter/tree.c @@ -0,0 +1,149 @@ +#include "tree_sitter/api.h" +#include "./array.h" +#include "./get_changed_ranges.h" +#include "./subtree.h" +#include "./tree_cursor.h" +#include "./tree.h" + +static const unsigned PARENT_CACHE_CAPACITY = 32; + +TSTree *ts_tree_new( +  Subtree root, const TSLanguage *language, +  const TSRange *included_ranges, unsigned included_range_count +) { +  TSTree *result = ts_malloc(sizeof(TSTree)); +  result->root = root; +  result->language = language; +  result->parent_cache = NULL; +  result->parent_cache_start = 0; +  result->parent_cache_size = 0; +  result->included_ranges = ts_calloc(included_range_count, sizeof(TSRange)); +  memcpy(result->included_ranges, included_ranges, included_range_count * sizeof(TSRange)); +  result->included_range_count = included_range_count; +  return result; +} + +TSTree *ts_tree_copy(const TSTree *self) { +  ts_subtree_retain(self->root); +  return ts_tree_new(self->root, self->language, self->included_ranges, self->included_range_count); +} + +void ts_tree_delete(TSTree *self) { +  if (!self) return; + +  SubtreePool pool = ts_subtree_pool_new(0); +  ts_subtree_release(&pool, self->root); +  ts_subtree_pool_delete(&pool); +  ts_free(self->included_ranges); +  if (self->parent_cache) ts_free(self->parent_cache); +  ts_free(self); +} + +TSNode ts_tree_root_node(const TSTree *self) { +  return ts_node_new(self, &self->root, ts_subtree_padding(self->root), 0); +} + +const TSLanguage *ts_tree_language(const TSTree *self) { +  return self->language; +} + +void ts_tree_edit(TSTree *self, const TSInputEdit *edit) { +  for (unsigned i = 0; i < self->included_range_count; i++) { +    TSRange *range = &self->included_ranges[i]; +    if (range->end_byte >= edit->old_end_byte) { +      if (range->end_byte != UINT32_MAX) { +        range->end_byte = edit->new_end_byte + (range->end_byte - edit->old_end_byte); +        range->end_point = point_add( +          edit->new_end_point, +          point_sub(range->end_point, edit->old_end_point) +        ); +        if (range->end_byte < edit->new_end_byte) { +          range->end_byte = UINT32_MAX; +          range->end_point = POINT_MAX; +        } +      } +      if (range->start_byte >= edit->old_end_byte) { +        range->start_byte = edit->new_end_byte + (range->start_byte - edit->old_end_byte); +        range->start_point = point_add( +          edit->new_end_point, +          point_sub(range->start_point, edit->old_end_point) +        ); +        if (range->start_byte < edit->new_end_byte) { +          range->start_byte = UINT32_MAX; +          range->start_point = POINT_MAX; +        } +      } +    } +  } + +  SubtreePool pool = ts_subtree_pool_new(0); +  self->root = ts_subtree_edit(self->root, edit, &pool); +  self->parent_cache_start = 0; +  self->parent_cache_size = 0; +  ts_subtree_pool_delete(&pool); +} + +TSRange *ts_tree_get_changed_ranges(const TSTree *self, const TSTree *other, uint32_t *count) { +  TSRange *result; +  TreeCursor cursor1 = {NULL, array_new()}; +  TreeCursor cursor2 = {NULL, array_new()}; +  TSNode root = ts_tree_root_node(self); +  ts_tree_cursor_init(&cursor1, root); +  ts_tree_cursor_init(&cursor2, root); + +  TSRangeArray included_range_differences = array_new(); +  ts_range_array_get_changed_ranges( +    self->included_ranges, self->included_range_count, +    other->included_ranges, other->included_range_count, +    &included_range_differences +  ); + +  *count = ts_subtree_get_changed_ranges( +    &self->root, &other->root, &cursor1, &cursor2, +    self->language, &included_range_differences, &result +  ); + +  array_delete(&included_range_differences); +  array_delete(&cursor1.stack); +  array_delete(&cursor2.stack); +  return result; +} + +void ts_tree_print_dot_graph(const TSTree *self, FILE *file) { +  ts_subtree_print_dot_graph(self->root, self->language, file); +} + +TSNode ts_tree_get_cached_parent(const TSTree *self, const TSNode *node) { +  for (uint32_t i = 0; i < self->parent_cache_size; i++) { +    uint32_t index = (self->parent_cache_start + i) % PARENT_CACHE_CAPACITY; +    ParentCacheEntry *entry = &self->parent_cache[index]; +    if (entry->child == node->id) { +      return ts_node_new(self, entry->parent, entry->position, entry->alias_symbol); +    } +  } +  return ts_node_new(NULL, NULL, length_zero(), 0); +} + +void ts_tree_set_cached_parent(const TSTree *_self, const TSNode *node, const TSNode *parent) { +  TSTree *self = (TSTree *)_self; +  if (!self->parent_cache) { +    self->parent_cache = ts_calloc(PARENT_CACHE_CAPACITY, sizeof(ParentCacheEntry)); +  } + +  uint32_t index = (self->parent_cache_start + self->parent_cache_size) % PARENT_CACHE_CAPACITY; +  self->parent_cache[index] = (ParentCacheEntry) { +    .child = node->id, +    .parent = (const Subtree *)parent->id, +    .position = { +      parent->context[0], +      {parent->context[1], parent->context[2]} +    }, +    .alias_symbol = parent->context[3], +  }; + +  if (self->parent_cache_size == PARENT_CACHE_CAPACITY) { +    self->parent_cache_start++; +  } else { +    self->parent_cache_size++; +  } +} | 
