• Steven Rostedt (VMware)'s avatar
    tracing: Fix histogram code when expression has same var as value · ce28d664
    Steven Rostedt (VMware) authored
    commit 8bcebc77 upstream.
    
    While working on a tool to convert SQL syntex into the histogram language of
    the kernel, I discovered the following bug:
    
     # echo 'first u64 start_time u64 end_time pid_t pid u64 delta' >> synthetic_events
     # echo 'hist:keys=pid:start=common_timestamp' > events/sched/sched_waking/trigger
     # echo 'hist:keys=next_pid:delta=common_timestamp-$start,start2=$start:onmatch(sched.sched_waking).trace(first,$start2,common_timestamp,next_pid,$delta)' > events/sched/sched_switch/trigger
    
    Would not display any histograms in the sched_switch histogram side.
    
    But if I were to swap the location of
    
      "delta=common_timestamp-$start" with "start2=$start"
    
    Such that the last line had:
    
     # echo 'hist:keys=next_pid:start2=$start,delta=common_timestamp-$start:onmatch(sched.sched_waking).trace(first,$start2,common_timestamp,next_pid,$delta)' > events/sched/sched_switch/trigger
    
    The histogram works as expected.
    
    What I found out is that the expressions clear out the value once it is
    resolved. As the variables are resolved in the order listed, when
    processing:
    
      delta=common_timestamp-$start
    
    The $start is cleared. When it gets to "start2=$start", it errors out with
    "unresolved symbol" (which is silent as this happens at the location of the
    trace), and the histogram is dropped.
    
    When processing the histogram for variable references, instead of adding a
    new reference for a variable used twice, use the same reference. That way,
    not only is it more efficient, but the order will no longer matter in
    processing of the variables.
    
    From Tom Zanussi:
    
     "Just to clarify some more about what the problem was is that without
      your patch, we would have two separate references to the same variable,
      and during resolve_var_refs(), they'd both want to be resolved
      separately, so in this case, since the first reference to start wasn't
      part of an expression, it wouldn't get the read-once flag set, so would
      be read normally, and then the second reference would do the read-once
      read and also be read but using read-once.  So everything worked and
      you didn't see a problem:
    
       from: start2=$start,delta=common_timestamp-$start
    
      In the second case, when you switched them around, the first reference
      would be resolved by doing the read-once, and following that the second
      reference would try to resolve and see that the variable had already
      been read, so failed as unset, which caused it to short-circuit out and
      not do the trigger action to generate the synthetic event:
    
       to: delta=common_timestamp-$start,start2=$start
    
      With your patch, we only have the single resolution which happens
      correctly the one time it's resolved, so this can't happen."
    
    Link: https://lore.kernel.org/r/20200116154216.58ca08eb@gandalf.local.home
    
    Cc: stable@vger.kernel.org
    Fixes: 067fe038 ("tracing: Add variable reference handling to hist triggers")
    Reviewed-by: default avatarTom Zanuss <zanussi@kernel.org>
    Tested-by: default avatarTom Zanussi <zanussi@kernel.org>
    Signed-off-by: default avatarSteven Rostedt (VMware) <rostedt@goodmis.org>
    Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
    ce28d664
trace_events_hist.c 136 KB