__entry->rcuname, __entry->p, __entry->nr_records)
 );
 
+/*
+ * Tracepoint for a normal synchronize_rcu() states. The first argument
+ * is the RCU flavor, the second argument is a pointer to rcu_head the
+ * last one is an event.
+ */
+TRACE_EVENT_RCU(rcu_sr_normal,
+
+       TP_PROTO(const char *rcuname, struct rcu_head *rhp, const char *srevent),
+
+       TP_ARGS(rcuname, rhp, srevent),
+
+       TP_STRUCT__entry(
+               __field(const char *, rcuname)
+               __field(void *, rhp)
+               __field(const char *, srevent)
+       ),
+
+       TP_fast_assign(
+               __entry->rcuname = rcuname;
+               __entry->rhp = rhp;
+               __entry->srevent = srevent;
+       ),
+
+       TP_printk("%s rhp=0x%p event=%s",
+               __entry->rcuname, __entry->rhp, __entry->srevent)
+);
+
 /*
  * Tracepoint for exiting rcu_do_batch after RCU callbacks have been
  * invoked.  The first argument is the name of the RCU flavor,
 
 {
        struct rcu_synchronize rs;
 
+       trace_rcu_sr_normal(rcu_state.name, &rs.head, TPS("request"));
+
        if (!READ_ONCE(rcu_normal_wake_from_gp)) {
                wait_rcu_gp(call_rcu_hurry);
-               return;
+               goto trace_complete_out;
        }
 
        init_rcu_head_on_stack(&rs.head);
        /* Now we can wait. */
        wait_for_completion(&rs.completion);
        destroy_rcu_head_on_stack(&rs.head);
+
+trace_complete_out:
+       trace_rcu_sr_normal(rcu_state.name, &rs.head, TPS("complete"));
 }
 
 /**