refactor(api): use arena for hlstate inspection

This commit is contained in:
bfredl 2024-02-09 12:41:14 +01:00
parent e9510211f0
commit 4cd43933d3
2 changed files with 40 additions and 24 deletions

View File

@ -2005,7 +2005,7 @@ Array nvim__inspect_cell(Integer grid, Integer row, Integer col, Arena *arena, E
ADD_C(ret, DICTIONARY_OBJ(hl_get_attr_by_id(attr, true, arena, err))); ADD_C(ret, DICTIONARY_OBJ(hl_get_attr_by_id(attr, true, arena, err)));
// will not work first time // will not work first time
if (!highlight_use_hlstate()) { if (!highlight_use_hlstate()) {
ADD_C(ret, ARRAY_OBJ(hl_inspect(attr))); ADD_C(ret, ARRAY_OBJ(hl_inspect(attr, arena)));
} }
return ret; return ret;
} }

View File

@ -116,12 +116,13 @@ retry: {}
// new attr id, send event to remote ui:s // new attr id, send event to remote ui:s
int id = (int)k; int id = (int)k;
Array inspect = hl_inspect(id); Arena arena = ARENA_EMPTY;
Array inspect = hl_inspect(id, &arena);
// Note: internally we don't distinguish between cterm and rgb attributes, // Note: internally we don't distinguish between cterm and rgb attributes,
// remote_ui_hl_attr_define will however. // remote_ui_hl_attr_define will however.
ui_call_hl_attr_define(id, entry.attr, entry.attr, inspect); ui_call_hl_attr_define(id, entry.attr, entry.attr, inspect);
api_free_array(inspect); arena_mem_free(arena_finish(&arena));
return id; return id;
} }
@ -129,10 +130,11 @@ retry: {}
void ui_send_all_hls(UI *ui) void ui_send_all_hls(UI *ui)
{ {
for (size_t i = 1; i < set_size(&attr_entries); i++) { for (size_t i = 1; i < set_size(&attr_entries); i++) {
Array inspect = hl_inspect((int)i); Arena arena = ARENA_EMPTY;
Array inspect = hl_inspect((int)i, &arena);
HlAttrs attr = attr_entry(i).attr; HlAttrs attr = attr_entry(i).attr;
remote_ui_hl_attr_define(ui, (Integer)i, attr, attr, inspect); remote_ui_hl_attr_define(ui, (Integer)i, attr, attr, inspect);
api_free_array(inspect); arena_mem_free(arena_finish(&arena));
} }
for (size_t hlf = 0; hlf < HLF_COUNT; hlf++) { for (size_t hlf = 0; hlf < HLF_COUNT; hlf++) {
remote_ui_hl_group_set(ui, cstr_as_string(hlf_names[hlf]), remote_ui_hl_group_set(ui, cstr_as_string(hlf_names[hlf]),
@ -204,7 +206,7 @@ int ns_get_hl(NS *ns_hl, int hl_id, bool link, bool nodefault)
if (!valid_item && p->hl_def != LUA_NOREF && !recursive) { if (!valid_item && p->hl_def != LUA_NOREF && !recursive) {
MAXSIZE_TEMP_ARRAY(args, 3); MAXSIZE_TEMP_ARRAY(args, 3);
ADD_C(args, INTEGER_OBJ((Integer)ns_id)); ADD_C(args, INTEGER_OBJ((Integer)ns_id));
ADD_C(args, CSTR_TO_OBJ(syn_id2name(hl_id))); ADD_C(args, CSTR_AS_OBJ(syn_id2name(hl_id)));
ADD_C(args, BOOLEAN_OBJ(link)); ADD_C(args, BOOLEAN_OBJ(link));
// TODO(bfredl): preload the "global" attr dict? // TODO(bfredl): preload the "global" attr dict?
@ -1176,17 +1178,30 @@ int object_to_color(Object val, char *key, bool rgb, Error *err)
} }
} }
Array hl_inspect(int attr) Array hl_inspect(int attr, Arena *arena)
{ {
// TODO(bfredl): use arena allocation if (!hlstate_active) {
Array ret = ARRAY_DICT_INIT; return (Array)ARRAY_DICT_INIT;
if (hlstate_active) {
hl_inspect_impl(&ret, attr);
} }
Array ret = arena_array(arena, hl_inspect_size(attr));
hl_inspect_impl(&ret, attr, arena);
return ret; return ret;
} }
static void hl_inspect_impl(Array *arr, int attr) static size_t hl_inspect_size(int attr)
{
if (attr <= 0 || attr >= (int)set_size(&attr_entries)) {
return 0;
}
HlEntry e = attr_entry(attr);
if (e.kind == kHlCombine || e.kind == kHlBlend || e.kind == kHlBlendThrough) {
return hl_inspect_size(e.id1) + hl_inspect_size(e.id2);
}
return 1;
}
static void hl_inspect_impl(Array *arr, int attr, Arena *arena)
{ {
Dictionary item = ARRAY_DICT_INIT; Dictionary item = ARRAY_DICT_INIT;
if (attr <= 0 || attr >= (int)set_size(&attr_entries)) { if (attr <= 0 || attr >= (int)set_size(&attr_entries)) {
@ -1196,35 +1211,36 @@ static void hl_inspect_impl(Array *arr, int attr)
HlEntry e = attr_entry(attr); HlEntry e = attr_entry(attr);
switch (e.kind) { switch (e.kind) {
case kHlSyntax: case kHlSyntax:
PUT(item, "kind", CSTR_TO_OBJ("syntax")); item = arena_dict(arena, 3);
PUT(item, "hi_name", PUT_C(item, "kind", CSTR_AS_OBJ("syntax"));
CSTR_TO_OBJ(syn_id2name(e.id1))); PUT_C(item, "hi_name", CSTR_AS_OBJ(syn_id2name(e.id1)));
break; break;
case kHlUI: case kHlUI:
PUT(item, "kind", CSTR_TO_OBJ("ui")); item = arena_dict(arena, 4);
PUT_C(item, "kind", CSTR_AS_OBJ("ui"));
const char *ui_name = (e.id1 == -1) ? "Normal" : hlf_names[e.id1]; const char *ui_name = (e.id1 == -1) ? "Normal" : hlf_names[e.id1];
PUT(item, "ui_name", CSTR_TO_OBJ(ui_name)); PUT_C(item, "ui_name", CSTR_AS_OBJ(ui_name));
PUT(item, "hi_name", PUT_C(item, "hi_name", CSTR_AS_OBJ(syn_id2name(e.id2)));
CSTR_TO_OBJ(syn_id2name(e.id2)));
break; break;
case kHlTerminal: case kHlTerminal:
PUT(item, "kind", CSTR_TO_OBJ("term")); item = arena_dict(arena, 2);
PUT_C(item, "kind", CSTR_AS_OBJ("term"));
break; break;
case kHlCombine: case kHlCombine:
case kHlBlend: case kHlBlend:
case kHlBlendThrough: case kHlBlendThrough:
// attribute combination is associative, so flatten to an array // attribute combination is associative, so flatten to an array
hl_inspect_impl(arr, e.id1); hl_inspect_impl(arr, e.id1, arena);
hl_inspect_impl(arr, e.id2); hl_inspect_impl(arr, e.id2, arena);
return; return;
case kHlUnknown: case kHlUnknown:
case kHlInvalid: case kHlInvalid:
return; return;
} }
PUT(item, "id", INTEGER_OBJ(attr)); PUT_C(item, "id", INTEGER_OBJ(attr));
ADD(*arr, DICTIONARY_OBJ(item)); ADD_C(*arr, DICTIONARY_OBJ(item));
} }