Profile.lua

Introduction

profile.lua is a small module used for finding bottlenecks in your Lua code. To use the profiler, you need to require the profile.lua file and specify when to start or stop collecting data. Once you are done profiling, a report is generated, describing:
  • which functions were called most frequently and
  • how much time was spent executing each function

    The source code is available on GitHub and the documentation is hosted on 2dengine.com

    Compatibility

    LuaJIT optimizations need to be disabled when using the profiler and co-routines are not supported. The profiler has been tested with both LuaJIT 2.0.5 and Lua 5.3 although there are no guarantees regarding its accuracy. The Lua debug module adds a tiny bit of overhead during each function call so recursion may produce worse results compared to using loops. Use at your own discretion!

    Examples

    Pure Lua

    local profile = require("profile")
    profile.start()
    -- execute code that will be profiled
    function isprime(n)
      for i = 2, math.sqrt(n) do
        if n%i == 0 then
          return false
        end
      end
      return true
    end
    for i = 1, 100000 do
      isprime(i)
    end
    profile.stop()
    -- report for the top 10 functions, sorted by execution time
    print(profile.report(10))
    

    LÖVE

    -- setup
    function love.load()
      love.profiler = require('profile') 
      love.profiler.start()
    end
    -- generates a report every 100 frames love.frame = 0 function love.update(dt) love.frame = love.frame + 1 if love.frame%100 == 0 then love.report = love.profiler.report(20) love.profiler.reset() end end
    -- prints the report function love.draw() love.graphics.print(love.report or "Please wait...") end

    Reports

    The default report is in plain text:
     +-----+----------------------------------+----------+--------------------------+----------------------------------+
     | #   | Function                         | Calls    | Time                     | Code                             |
     +-----+----------------------------------+----------+--------------------------+----------------------------------+
     | 1   | update                           | 1        | 9.0023296745494          | main.lua:23                      |
     | 2   | f                                | 1        | 9.0022503120126          | main.lua:12                      |
     | 3   | g                                | 8        | 8.0016986143455          | main.lua:5                       |
     | 4   | [string "boot.lua"]:185          | 3        | 2.4960798327811e-005     | [string "boot.lua"]:185          |
     | 5   | [string "boot.lua"]:134          | 2        | 1.7920567188412e-005     | [string "boot.lua"]:134          |
     | 6   | [string "boot.lua"]:188          | 1        | 1.6000514733605e-005     | [string "boot.lua"]:188          |
     | 7   | [string "boot.lua"]:182          | 1        | 1.2160395272076e-005     | [string "boot.lua"]:182          |
     | 8   | [string "boot.lua"]:131          | 1        | 1.0240328265354e-005     | [string "boot.lua"]:131          |
     | 9   | load                             | 0        | 0                        | main.lua:17                      |
     +-----+----------------------------------+----------+--------------------------+----------------------------------+
    

    The "Function" column shows the function name. Closures that do not have a defined name will be replaced with the source code file and line number. The "Calls" column tracks the total number of function calls. The "Time" column records the total execution time. This value is usually in seconds. The "Code" column shows the source file and the line number. Chunks executed using loadstring are labeled as "string". It's easy to generate reports in a custom format, for example CSV:
    print('Position,Function name,Number of calls,Time,Source,')
    for t in ipairs(profiler.query(10)) do
      print(table.concat(t, ",")..",")
    end
    

    Credits

    0x25a0 grump Roland Yonaba
    Please support our work so we can release more free software in the future.

  • Profile.lua > profile

    profile

    The "profile" module controls when to start or stop collecting data and can be used to generate reports.
    profile.comp(a, b)
    profile.hooker(event, line, info)
    profile.query(limit)
    profile.report(limit)
    profile.reset()
    profile.setclock(func)
    profile.start()
    profile.stop()

    profile.comp(a, b)

    This is an internal function.
    Arguments
    functiona
    First function
    functionb
    Second function
    Returns
    boolean True if "a" should rank higher than "b"

    profile.hooker(event, line, info)

    This is an internal function.
    Arguments
    stringevent
    Event type
    numberline
    Line number
    tableinfo (optional)
    Debug info table

    profile.query(limit)

    Generates a report of functions that have been called since the profile was started. Returns the report as a numeric table of rows containing the rank, function label, number of calls, total execution time and source code line number.
    Arguments
    numberlimit (optional)
    Maximum number of rows
    Returns
    table Table of rows

    profile.report(limit)

    Generates a text report of functions that have been called since the profile was started. Returns the report as a string that can be printed to the console.
    Arguments
    numberlimit (optional)
    Maximum number of rows
    Returns
    string Text-based profiling report

    profile.reset()

    Resets all collected data.

    profile.setclock(func)

    Sets a clock function to be used by the profiler.
    Arguments
    functionfunc
    Clock function that returns a number

    profile.start()

    Starts collecting data.

    profile.stop()

    Stops collecting data.

    This document was last updated on January 20, 2025, by 2dengine LLC. © All rights reserved.