2019-01-10 15:22:39 -08:00
|
|
|
#include "tree_sitter/api.h"
|
2019-01-04 17:33:34 -08:00
|
|
|
#include "./array.h"
|
|
|
|
|
#include "./get_changed_ranges.h"
|
2022-08-18 13:48:47 -07:00
|
|
|
#include "./length.h"
|
2019-01-04 17:33:34 -08:00
|
|
|
#include "./subtree.h"
|
|
|
|
|
#include "./tree_cursor.h"
|
|
|
|
|
#include "./tree.h"
|
2018-05-10 22:22:37 -07:00
|
|
|
|
2018-11-07 12:56:44 -08:00
|
|
|
TSTree *ts_tree_new(
|
|
|
|
|
Subtree root, const TSLanguage *language,
|
|
|
|
|
const TSRange *included_ranges, unsigned included_range_count
|
|
|
|
|
) {
|
2018-05-10 22:22:37 -07:00
|
|
|
TSTree *result = ts_malloc(sizeof(TSTree));
|
|
|
|
|
result->root = root;
|
|
|
|
|
result->language = language;
|
2018-11-07 12:56:44 -08:00
|
|
|
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;
|
2018-05-10 22:22:37 -07:00
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TSTree *ts_tree_copy(const TSTree *self) {
|
|
|
|
|
ts_subtree_retain(self->root);
|
2018-11-07 12:56:44 -08:00
|
|
|
return ts_tree_new(self->root, self->language, self->included_ranges, self->included_range_count);
|
2018-05-10 22:22:37 -07:00
|
|
|
}
|
|
|
|
|
|
2018-05-11 15:06:13 -07:00
|
|
|
void ts_tree_delete(TSTree *self) {
|
2019-05-30 11:47:56 -04:00
|
|
|
if (!self) return;
|
|
|
|
|
|
2018-05-10 22:22:37 -07:00
|
|
|
SubtreePool pool = ts_subtree_pool_new(0);
|
|
|
|
|
ts_subtree_release(&pool, self->root);
|
|
|
|
|
ts_subtree_pool_delete(&pool);
|
2018-11-07 12:56:44 -08:00
|
|
|
ts_free(self->included_ranges);
|
2018-05-10 22:22:37 -07:00
|
|
|
ts_free(self);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TSNode ts_tree_root_node(const TSTree *self) {
|
2018-09-17 13:12:13 -07:00
|
|
|
return ts_node_new(self, &self->root, ts_subtree_padding(self->root), 0);
|
2018-05-10 22:22:37 -07:00
|
|
|
}
|
|
|
|
|
|
2022-08-18 13:48:47 -07:00
|
|
|
TSNode ts_tree_root_node_with_offset(
|
|
|
|
|
const TSTree *self,
|
|
|
|
|
uint32_t offset_bytes,
|
|
|
|
|
TSPoint offset_extent
|
|
|
|
|
) {
|
|
|
|
|
Length offset = {offset_bytes, offset_extent};
|
|
|
|
|
return ts_node_new(self, &self->root, length_add(offset, ts_subtree_padding(self->root)), 0);
|
|
|
|
|
}
|
|
|
|
|
|
2018-06-21 12:54:19 -07:00
|
|
|
const TSLanguage *ts_tree_language(const TSTree *self) {
|
|
|
|
|
return self->language;
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-10 22:22:37 -07:00
|
|
|
void ts_tree_edit(TSTree *self, const TSInputEdit *edit) {
|
2018-11-07 12:56:44 -08:00
|
|
|
for (unsigned i = 0; i < self->included_range_count; i++) {
|
|
|
|
|
TSRange *range = &self->included_ranges[i];
|
|
|
|
|
if (range->end_byte >= edit->old_end_byte) {
|
2018-11-14 11:28:43 -08:00
|
|
|
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;
|
|
|
|
|
}
|
2018-11-07 12:56:44 -08:00
|
|
|
}
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-10 22:22:37 -07:00
|
|
|
SubtreePool pool = ts_subtree_pool_new(0);
|
|
|
|
|
self->root = ts_subtree_edit(self->root, edit, &pool);
|
2018-05-11 16:53:47 -07:00
|
|
|
ts_subtree_pool_delete(&pool);
|
2018-05-10 22:22:37 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TSRange *ts_tree_get_changed_ranges(const TSTree *self, const TSTree *other, uint32_t *count) {
|
2018-11-07 15:11:21 -08:00
|
|
|
TreeCursor cursor1 = {NULL, array_new()};
|
|
|
|
|
TreeCursor cursor2 = {NULL, array_new()};
|
2019-10-28 15:32:52 -07:00
|
|
|
ts_tree_cursor_init(&cursor1, ts_tree_root_node(self));
|
|
|
|
|
ts_tree_cursor_init(&cursor2, ts_tree_root_node(other));
|
2018-11-07 12:56:44 -08:00
|
|
|
|
|
|
|
|
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
|
|
|
|
|
);
|
|
|
|
|
|
2019-10-28 15:32:52 -07:00
|
|
|
TSRange *result;
|
2018-05-10 22:22:37 -07:00
|
|
|
*count = ts_subtree_get_changed_ranges(
|
2018-09-17 13:12:13 -07:00
|
|
|
&self->root, &other->root, &cursor1, &cursor2,
|
2018-11-07 12:56:44 -08:00
|
|
|
self->language, &included_range_differences, &result
|
2018-05-10 22:22:37 -07:00
|
|
|
);
|
2018-11-07 12:56:44 -08:00
|
|
|
|
|
|
|
|
array_delete(&included_range_differences);
|
2018-05-10 22:22:37 -07:00
|
|
|
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);
|
|
|
|
|
}
|