----- On Nov 16, 2020, at 5:51 PM, rostedt rostedt@xxxxxxxxxxx wrote: > [ Kees, I added you because you tend to know about these things. > Is it OK to assign a void func(void) that doesn't do anything and returns > nothing to a function pointer that could be call with parameters? We need > to add stubs for tracepoints when we fail to allocate a new array on > removal of a callback, but the callbacks do have arguments, but the stub > called does not have arguments. [...] > +/* Called in removal of a func but failed to allocate a new tp_funcs */ > +static void tp_stub_func(void) > +{ > + return; > +} > + In C, the "void" unnamed function parameter specifies that the function has no parameters. C99 section 6.7.5.3 Function declarators (including prototypes): "The special case of an unnamed parameter of type void as the only item in the list specifies that the function has no parameters." The C99 standard section "6.5.2.2 Function calls" states: "If the function is defined with a type that is not compatible with the type (of the expression) pointed to by the expression that denotes the called function, the behavior is undefined." "J.2 Undefined behavior" states: "For a call to a function without a function prototype in scope, the number of arguments does not equal the number of parameters (6.5.2.2)." I suspect that calling a function expecting no parameters from a call site with parameters might work for the cdecl calling convention because the caller is responsible for stack cleanup, but it seems to rely on a behavior which is very much tied to the calling convention, and within "undefined behavior" territory as far as the C standard is concerned. I checked whether going for "void tp_stub_func()" instead would work better, but it seems to also mean "no parameter" when applied to the function definition. It's only when applied on the function declarator that is not part of the definition that it means no information about the number or type of parameters is supplied. (see C99 "6.7.5.3 Function declarators (including prototypes)" items 14 and 15) And the kernel builds with "-Werror=strict-prototypes" which would not allow it anyway. One thing which would work with respect to the C standard is to define one stub function per tracepoint. This add 16 bytes of code per TRACE_DEFINE() on x86-64, but by specifying that those are cache-cold with "__cold", I notice that it adds no extra code size my build of kernel/sched/core.o which contains 30 tracepoint definitions. diff --git a/include/linux/tracepoint-defs.h b/include/linux/tracepoint-defs.h index e7c2276be33e..e0351bb0b140 100644 --- a/include/linux/tracepoint-defs.h +++ b/include/linux/tracepoint-defs.h @@ -38,6 +38,7 @@ struct tracepoint { int (*regfunc)(void); void (*unregfunc)(void); struct tracepoint_func __rcu *funcs; + void *stub_func; }; #ifdef CONFIG_HAVE_ARCH_PREL32_RELOCATIONS diff --git a/include/linux/tracepoint.h b/include/linux/tracepoint.h index 0f21617f1a66..b0b805de3779 100644 --- a/include/linux/tracepoint.h +++ b/include/linux/tracepoint.h @@ -287,6 +287,9 @@ static inline struct tracepoint *tracepoint_ptr_deref(tracepoint_ptr_t *p) #define DEFINE_TRACE_FN(_name, _reg, _unreg, proto, args) \ static const char __tpstrtab_##_name[] \ __section("__tracepoints_strings") = #_name; \ + static void __cold __tracepoint_stub_func_##_name(void *__data, proto) \ + { \ + } \ extern struct static_call_key STATIC_CALL_KEY(tp_func_##_name); \ int __traceiter_##_name(void *__data, proto); \ struct tracepoint __tracepoint_##_name __used \ @@ -298,7 +301,8 @@ static inline struct tracepoint *tracepoint_ptr_deref(tracepoint_ptr_t *p) .iterator = &__traceiter_##_name, \ .regfunc = _reg, \ .unregfunc = _unreg, \ - .funcs = NULL }; \ + .funcs = NULL, \ + .stub_func = __tracepoint_stub_func_##_name, }; \ __TRACEPOINT_ENTRY(_name); \ int __traceiter_##_name(void *__data, proto) \ { \ Thanks, Mathieu -- Mathieu Desnoyers EfficiOS Inc. http://www.efficios.com