1
0
Fork 0
mirror of https://github.com/ganelson/inform.git synced 2024-07-08 18:14:21 +03:00
inform7/inter/pipeline-module/Chapter 3/Detect Indirect Calls.w
2021-11-14 23:40:33 +00:00

56 lines
2.2 KiB
OpenEdge ABL

[DetectIndirectCalls::] Detect Indirect Calls.
To make sure certain symbol names translate into globally unique target symbols.
@h Pipeline stage.
=
void DetectIndirectCalls::create_pipeline_stage(void) {
ParsingPipelines::new_stage(I"detect-indirect-calls", DetectIndirectCalls::run_pipeline_stage, NO_STAGE_ARG, FALSE);
}
int DetectIndirectCalls::run_pipeline_stage(pipeline_step *step) {
InterTree::traverse(step->ephemera.repository, DetectIndirectCalls::visitor, step, NULL, PACKAGE_IST);
return TRUE;
}
void DetectIndirectCalls::visitor(inter_tree *I, inter_tree_node *P, void *state) {
pipeline_step *step = (pipeline_step *) state;
inter_package *pack = Inter::Package::defined_by_frame(P);
if (Inter::Packages::is_codelike(pack)) {
inter_tree_node *D = Inter::Packages::definition(pack);
DetectIndirectCalls::traverse_code_tree(D, step);
}
}
@ |iden[0]| and |iden[1]| hold left and right identity elements for these binary
operations:
=
void DetectIndirectCalls::traverse_code_tree(inter_tree_node *P, pipeline_step *step) {
PROTECTED_LOOP_THROUGH_INTER_CHILDREN(F, P) {
DetectIndirectCalls::traverse_code_tree(F, step);
}
PROTECTED_LOOP_THROUGH_INTER_CHILDREN(F, P) {
if ((F->W.data[ID_IFLD] == INV_IST) &&
(F->W.data[METHOD_INV_IFLD] == INVOKED_ROUTINE)) {
inter_symbol *routine = InterSymbolsTables::symbol_from_frame_data(F, INVOKEE_INV_IFLD);
if (routine == NULL) internal_error("bad routine");
inter_tree_node *D = routine->definition;
if ((D) && (D->W.data[ID_IFLD] == VARIABLE_IST)) {
inter_tree *I = F->tree;
F->W.data[METHOD_INV_IFLD] = INVOKED_PRIMITIVE;
int arity = 0;
PROTECTED_LOOP_THROUGH_INTER_CHILDREN(X, F) arity++;
inter_ti bip = Primitives::indirect_interp(arity);
inter_symbol *prim_symb = Primitives::get(I, bip);
F->W.data[INVOKEE_INV_IFLD] = InterSymbolsTables::id_from_symbol_F(F, NULL, prim_symb);
inter_bookmark IBM = Inter::Bookmarks::first_child_of_this_node(I, F);
inter_ti val1 = 0, val2 = 0;
Inter::Symbols::to_data(Inter::Bookmarks::tree(&IBM), Inter::Bookmarks::package(&IBM), routine, &val1, &val2);
Inter::Val::new(&IBM, RunningPipelines::get_symbol(step, unchecked_kind_RPSYM), (int) F->W.data[LEVEL_IFLD] + 1, val1, val2, NULL);
}
}
}
}