<div dir="ltr">Hello,<div class="gmail_extra"><br><div class="gmail_quote">On Thu, Sep 24, 2015 at 11:58 PM, Philip Clarke <span dir="ltr"><<a href="mailto:send2philip@gmail.com" target="_blank">send2philip@gmail.com</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0px 0px 0px 0.8ex;border-left-width:1px;border-left-color:rgb(204,204,204);border-left-style:solid;padding-left:1ex"><div dir="ltr"><div>One common use case for tracing I would like is the ability to trace a function call for just the next request in the system.</div><div><br></div><div>Currently the PidSpec in erlang:trace/3 allows for pid, existing, new or all.</div><div><br></div><div>We would use pid if we knew the pid of the next short lived process to call a function (but we don't).</div><div>We cannot use new either as tracing a function call for all new processes might mean that this function is traced 1000s times per second.  </div><div><br></div><div>What I would like is some new PidSpec like 'next_pid' which would specify that I want to trace all new processes until the function call I am tracing has been triggered. Once the trace is triggered then it should turn off tracing on all processes.</div><div><br></div><div>e.g.</div><div>dbg:p(next_pid, call).</div><div>dbg:tp(M, F, x).</div><div><br></div><div>This would be a great help in my work where quite often I want to see the typical format of a request from a client and just capturing 1 request is enough. I believe that libraries such as redbug have this kind of ability, but perhaps the feature is fundamental enough to include as part of trace ?</div><div><br></div></div></blockquote><div><br></div><div>Doing this kind of thing and guaranteeing that just one trace will be printed cannot be done without suspending all schedulers running in the system, which is something that I'm pretty sure you do not want :) You can however come pretty close even today by doing something like this:</div><div><br></div><div><div>-module(single_trace).</div><div><br></div><div>-export([trace/2,trace/4]).</div><div><br></div><div>trace(M,F) -></div><div>  trace(M,F,'_',[]).</div><div>trace(M,F,A,MS) -></div><div>  DisableTrace = {'_',[],[{disable_trace,[call]}]},</div><div>  erlang:trace_pattern({M,F,A}, [ DisableTrace | MS], []),</div><div>  Tracer = spawn(fun trace_loop/0),</div><div>  erlang:trace(all, true, [{tracer, Tracer}, call]).</div><div><br></div><div>trace_loop() -></div><div>  receive</div><div>    Msg -></div><div>      io:format("~p~n",[Msg]),</div><div>      erlang:trace(all, false, [call]),</div><div>      trace_loop()</div><div>  end.</div></div><div><br></div><div><div>12> spawn(fun F() -> lists:map(fun(I) -> I + 1 end, lists:seq(1,100)), F() end). </div><div><0.53.0></div><div>13> spawn(fun F() -> lists:map(fun(I) -> I + 1 end, lists:seq(1,100)), F() end). </div><div><0.55.0></div><div>14> spawn(fun F() -> lists:map(fun(I) -> I + 1 end, lists:seq(1,100)), F() end). </div><div><0.57.0></div><div>15> spawn(fun F() -> lists:map(fun(I) -> I + 1 end, lists:seq(1,100)), F() end). </div><div><0.59.0></div><div>16> spawn(fun F() -> lists:map(fun(I) -> I + 1 end, lists:seq(1,100)), F() end). </div><div><0.61.0></div><div>17> single_trace:trace(lists,map).</div></div><div><div>{trace,<0.61.0>,call,</div><div>    {lists,map,</div><div>       [#Fun<erl_eval.6.54118792>,</div><div>        [1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,</div><div>        24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40,41,42,43,</div><div>        44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60,61,62,63,</div><div>        64,65,66,67,68,69,70,71,72,73,74,75,76,77,78,79,80,81,82,83,</div><div>        84,85,86,87,88,89,90,91,92,93,94,95,96,97,98,99,100]]}}</div></div><div>18></div><div><br></div><div>Writing your own little tracer process can sometimes do wonders :) This will occasionally print a couple of more messages, but it is quite fast at turning things off. </div><div><br></div><div>Lukas</div></div></div></div>