#help_index "Debugging/Profiler;Profiler;Cmd Line (Typically)/Profiler"
#help_file "::/Doc/Profiler"

#define PF_ARRAY_COUNT 0x100000
I64  pf_jiffy_start, pf_jiffy_end;
I64 *pf_array = NULL;
I64  pf_cpu = 0;
I64  pf_buf_in_ptr = 0, pf_depth;
I64  pf_prof_active = 0;

U0 ProfTimerInt(CTask *task)
{//See profiler_timer_irq.
    I64 i, k;

    if (Bt(&pf_prof_active, 0))
        for (k = 0; k <= pf_depth; k++)
        {
            if (task == Gs->idle_task)
                i = SYS_IDLE_PT;
            else
                i = TaskCaller(task, k, TRUE);
            if (pf_buf_in_ptr < PF_ARRAY_COUNT)
            {
                pf_array[pf_buf_in_ptr++] = i;
                pf_jiffy_end = counts.jiffies;
            }
        }
}

public U0 Prof(I64 depth=0, I64 cpu_num=0)
{/*Start collecting profiler statistics.
Profilers report where time is spent
by sampling RIP during the 1000Hz
timer interrupt.

Do a ProfRep(), (profiler report)
after you have collected data.
*/
    if (!(0 <= cpu_num < mp_count))
        ST_ERR_ST "Invalid CPU\n";
    else
    {
        cpu_structs[pf_cpu].profiler_timer_irq = NULL;
        pf_cpu = cpu_num;

        pf_depth = depth;
        pf_buf_in_ptr = 0;
        if (!pf_array)
            pf_array = ZMAlloc(sizeof(I64) * PF_ARRAY_COUNT);
        pf_jiffy_end = pf_jiffy_start = counts.jiffies;
        LBts(&pf_prof_active, 0);
        cpu_structs[pf_cpu].profiler_timer_irq = &ProfTimerInt;
    }
}

I64 ProfCompare(U8 *i1, U8 *i2)
{
    return i1 - i2;
}

public U0 ProfRep(I64 filter_count=1, Bool leave_it=OFF)
{//Profiler report. Call Prof() first and collect data.
    I64 i, hits, rip, last_rip = 0, routine_total = 0;
    F64 total_time;
    U8  buf[256], buf2[256], last_buf[256];

    if (!LBtr(&pf_prof_active, 0))
        "Profiler Not Active\n";
    if (!pf_buf_in_ptr)
        "No Profiler Statistic\n";
    else
    {
        if (!(total_time = pf_jiffy_end - pf_jiffy_start))
            total_time = 1;
        QuickSortI64(pf_array, pf_buf_in_ptr, &ProfCompare);
        *last_buf = 0;
        for (i = 0; i < pf_buf_in_ptr; i += hits)
        {
            rip = pf_array[i];
            hits = 0;
            do hits++;
            while (i + hits < pf_buf_in_ptr && pf_array[i + hits] == rip);

            StrPrint(buf, "%p", rip);
            StrFirstRemove(buf, "+", buf2);
            if (StrCompare(buf2, last_buf))
            {
                if (*last_buf && routine_total >= filter_count)
                    "$GREEN$%6.2f %08X:%s\n$FG$", 100 * routine_total / total_time, routine_total, last_buf;
                StrCopy(last_buf, buf2);
                routine_total = 0;
            }
            routine_total += hits;
            if (hits >= filter_count)
            {
                "%6.2f %08X:%P\n", 100 * hits / total_time, hits, rip;
                last_rip=rip;
            }
        }
        if (*last_buf && routine_total >= filter_count)
            "$GREEN$%6.2f %08X:%s\n$FG$", 100 * routine_total / total_time, routine_total, last_buf;
        "Total Time:%0.6fs\n", total_time / JIFFY_FREQ;
        if (leave_it)
        {
            cpu_structs[pf_cpu].profiler_timer_irq = &ProfTimerInt;
            LBts(&pf_prof_active, 0);
        }
        else
            cpu_structs[pf_cpu].profiler_timer_irq = NULL;
    }
}