Actually, another interesting hint is that the original foo function takes two parameters. The function I am seeing inside the FOR_EACH_FUNCTION_WITH_GIMPLE_BODY is a specialized function of foo with only 1 parameter. However, the indirect function call is a version of foo which has not been specialized (i.e. it takes the original two parameters).

I guess my questions would be:

* Does FOR_EACH_FUNCTION_WITH_GIMPLE_BODY only iterates across functions which are reachable for main across the call graph? * Is the the underlying mechanism for FOR_EACH_FUNCTION_WITH_GIMPLE_BODY not updated after ipa-prop discovers targets of indirect functions? * Or is it just that the new callsite does not have a gimple body for its function? (This seems implausible since the new direct callsite should refer to the original function implementation.) How can I access this function's body?

Thanks.


On 14/07/2020 12:37, Erick Ochoa wrote:
Hello,

I have a function foo defined on a source file. Sometimes, a function pointer pointing to foo is passed as a parameter to other functions where foo is called indirectly. This indirect call is specialized during link time. Still at link time, I analyze the function call the following way:

   // s is the gimple statement which corresponds to the indirect call
   tree fn = gimple_call_fndecl(s);
   // for this particular call the assertions are true
   gcc_assert(fn)
   cgraph_node *node = cgraph_node::get(fn)
   gcc_assert(node)

I have analyzed the body of this function previously by using the FOR_EACH_FUNCTION_WITH_GIMPLE_BODY macro. However, I do not know if there's a way to link the cnode_graph (or function decl) analyzed in the macro with the one obtained at the call site. What would be the best way to say something like:

   tree fn = gimple_call_fndecl(s);
   // for this particular call the assertions are true
   gcc_assert(fn)
   cgraph_node *node = cgraph_node::get(fn)
   gcc_assert(node)
   bool i_want_this_to_be_true = saw_fn_in_loop_before(node, fn);

I don't think that using storing the for loop and checking in saw_fn_in_loop_before is the way to go because I believe cnode->decl pointers can be different. Is this correct? In other words


FOR_EACH_FUNCTION_WITH_GIMPLE_BODY(cnode)
{
   a_std_set.insert(cnode->decl)
}

// later

bool
saw_fn_in_loop_before(cnode_graph *cn, tree fn)
{
   return a_std_set.find(fn) != a_std_set.end();
}

Should not work. Something I found interesting is that for the fndecl obtained through the callsite gimple_has_body_p returns false. Even though I saw a fndecl which corresponds to the same function in the FOR_EACH_FUNCTION_WITH_GIMPLE_BODY.

Thanks! Any hints are appreciated!

Reply via email to