Lua Plugins

libinput provides a plugin system that allows users to modify the behavior of devices. For example, a plugin may add or remove axes and/or buttons on a device and/or modify the event stream seen by this device before it is passed to libinput.

Plugins are implemented in Lua (version 5.4) and are typically loaded from the following paths:

  • /etc/libinput/plugins/*.lua, and

  • /usr/lib{64}/libinput/plugins/*.lua

Plugins are loaded in alphabetical order and where multiple plugins share the same file name, the one in the highest precedence directory is used. Plugins in /etc take precedence over plugins in /usr.

Note

Plugins lookup paths and their order are decided by the compositor. Some compositors may support more/fewer/other lookup paths than the above defaults.

Plugins are run sequentially in ascending sort-order (i.e. 00-foo.lua runs before 10-bar.lua) and each plugin sees the state left by any previous plugins. For example if 00-foo.lua changes all left button events to right button events, 10-bar.lua only ever sees right button events.

See the Lua Reference manual for details on the Lua language.

Note

Plugins are not loaded by default, it is up to the compositor whether to allow plugins. An explicit call to libinput_plugin_system_load_plugins() is required.

Limitations

Each script runs in its own sandbox and cannot communicate or share state with other scripts.

Tables that hold API methods are not writable, i.e. it is not possible to overwrite the default functionality of those APIs.

The Lua API available to plugins is limited to the following calls:

assert  error   ipairs  next     pairs  tonumber
pcall   select  print   tostring type   xpcall
table   string  math    _VERSION

It is not possible to e.g. use the io module from a script.

To use methods on instantiated objects, the object:method method call syntax must be used. For example:

libinput:register()
libinput.register() -- this will fail

When to use plugins

libinput plugins are a relatively niche use-case that typically need to address either once-off issues (e.g. those caused by worn-out hardware) or user preferences that libinput does not and will not cater for.

Plugins should not be used for issues that can be fixed generically, for example via Device quirks.

As a rule of thumb: a plugin should be a once-off that only works for one user’s hardware. If a plugin can be shared with many users then the plugin implements functionality that should be integrated into libinput proper.

Testing plugins

Our Helper tools support plugins if passed the --enable-plugins commandline option. For implementing and testing plugins the easiest commands to test are

Where libinput is built and run from git, the tools will also look for plugins in the meson build directory. See the plugins/meson.build file for details.

Lua Plugin API

Lua plugins sit effectively below libinput and the API is not a representation of the libinput API. Plugins modify the evdev event stream received from the kernel.

digraph stack
{
  compound=true;
  rankdir="LR";
  node [
    shape="box";
  ]

  subgraph cluster_2 {
	  label="Kernel";
	  event0 [label="/dev/input/event0"]
  }

  subgraph cluster_1 {
	  label="libinput";
    subgraph cluster_0 {
	    label="Plugin pipeline";
	    p1 [label="00-foo.lua"];
	    p2 [label="10-bar.lua"];
    }
	  libinput [label="libinput core"];
  }


  compositor [label="Compositor"];

  event0 -> p1;
  p1 -> p2;
  p2 -> libinput;
  libinput -> compositor [ltail=cluster_1 label="libinput API"];
}

The API revolves around two types: libinput and EvdevDevice. The libinput type is used to register a plugin from a script, the EvdevDevice represents one device that is present in the system (but may not have yet been added by libinput).

Typically a script does the following steps:

  • register with libinput via libinput:register({versions})

  • connect to the "new-evdev-device" event

  • receive an EvdevDevice object in the "new-evdev-device" callback

    • check and/or modify the evdev event codes on the device

    • connect to the device’s "evdev-frame" event

  • receive an evdev frame in the device’s "evdev-frame" callback

    • check and/or modify the events in that frame

Where multiple plugins are active, the evdev frame passed to the callback is the combined frame as processed by all previous plugins in ascending sort order. For example, if one plugin discards all button events subsequent plugins will never see those button events in the frame.

Plugin version stability

Plugin API version stability is provided on a best effort basis. We aim to provide stable plugin versions for as long as feasible but may need to retire some older versions over time. For this reason a plugin can select multiple versions it implements, libinput will pick one supported version and adjust the plugin behavior to match that version. See the libinput:register() call for details.

Lua Plugin API Reference

libinput provides the following globals and types:

Evdev Usages

Evdev usages are a libinput-specific wrapper around the linux/input-event-codes.h evdev types and codes. They are used by libinput internally and are a 32-bit combination of type << 16 | code. Each usage carries the type and code and is thus simpler to pass around and less prone to type confusion.

The evdev global attempts to provide all available usages but for the niche cases where it does not provide a named constant the value can be crafted manually:

evdev_type = 0x3  -- EV_REL
evdev_code = 0x1  -- REL_Y
evdev_usage = (evdev_type << 16) | evdev_code

assert(usage == evdev.REL_Y)

The evdev global

The evdev global represents all known Evdev Usages, effectively in the form:

evdev = {
   ABS_X = (3 << 16) | 0,
   ABS_Y = (3 << 16) | 1,
   ...
   REL_X = (2 << 16) | 0,
   REL_Y = (2 << 16) | 1,
   ...
}

This global is provided for convenience to improve readability in the code. Note that the name uses the event code name only (e.g. evdev.ABS_Y) but the value is an Evdev Usage (type and code).

See the linux/input-event-codes.h header file provided by your kernel for a list of all evdev types and codes.

The evdev global also provides the bus type constants, e.g. evdev.BUS_USB. See the linux/input.h header file provided by your kernel for a list of bus types.

Evdev frames

Evdev frames represent a single frame of evdev events for a device. A frame is a group of events that occurred at the same time. The frame usually only contains state that has changed compared to the previous frame.

In our API a frame is exposed as a nested table with the following structure:

frame1 = {
     { usage = evdev.ABS_X, value = 123 },
     { usage = evdev.ABS_Y, value = 456 },
     { usage = evdev.BTN_LEFT, value = 1 },
}
frame2 = {
     { usage = evdev.ABS_Y, value = 457 },
}
frame3 = {
     { usage = evdev.ABS_X, value = 124 },
     { usage = evdev.BTN_LEFT, value = 0 },
}

Note

This API does not use SYN_REPORT events, it is implied at the end of the table. Where a plugin writes a SYN_REPORT into the list of events, that SYN_REPORT terminates the event frame (similar to writing a \0 into the middle of a C string). A frame containing only a SYN_REPORT is functionally equivalent to an empty frame.

Events or frames do not have a timestamp. Where a timestamp is required, that timestamp is passed as additional argument to the function or return value.

See The evdev global for a list of known usages.

Warning

Evdev frames have an implementation-defined size limit of how many events can be added to a single frame. This limit should never be hit by valid plugins.

The libinput global object

The core of our plugin API is the libinput global object. A script must immediately register() to be active, otherwise it is unloaded immediately.

All libinput-specific APIs can be accessed through the libinput object.

libinput:register({1, 2, ...})

Register this plugin with the given table of supported version numbers and returns the version number selected by libinput for this plugin. See Plugin version stability for details.

-- this plugin can support versions 1, 4 and 5
version = libinput:register({1, 4, 5})
if version == 1 then
    ...

This function must be the first function called. If the plugin calls any other functions before register(), those functions return the default zero value for the return type (nil, 0, an empty table, etc.).

If the plugin does not call register() it will be removed immediately. Once registered, any connected callbacks will be invoked whenever libinput detects new devices, removes devices, etc.

This function must only be called once.

libinput:unregister()

Unregister this plugin. This removes the plugin from libinput and releases any resources associated with this plugin. This call must be the last call in your plugin, it is effectively equivalent to Lua’s os.exit().

libinput:log_debug(message)

Log a message at the libinput debug log priority. See libinput:log_error() for details.

libinput:log_info(message)

Log a message at the libinput info log priority. See libinput:log_error() for details.

libinput:log_error(message)

Log a message at the libinput error log priority. Whether a message is displayed in the log depends on libinput’s log priority, set by the caller.

A compositor may disable stdout and stderr. Log messages should be preferred over Lua’s print() function to ensure the messages end up in the same location as other libinput log messages and are not discarded.

libinput:now()

Returns the current time in microseconds in CLOCK_MONOTONIC. This is the timestamp libinput uses internally. This timestamp cannot be mapped to any particular time of day, see the clock_gettime() man page for details.

libinput:version()

Returns the agreed-on version of the plugin, see libinput:register(). If called before libinput:register() this function returns 0.

libinput:connect(name, function)

Set the callback to the given event name. Only one callback may be set for an event name at any time, subsequent callbacks will replace any earlier callbacks for the same name.

Version 1 of the plugin API supports the following events and callback arguments:

  • "new-evdev-device": A new EvdevDevice has been seen by libinput but not yet added.

    libinput:connect("new-evdev-device", function (device) ... end)
    
  • "timer-expired": The timer for this plugin has expired. This event is only sent if the plugin has set a timer with timer_set().

    libinput:connect("timer-expired", function (now) ... end)
    

    The now argument is the current time in microseconds in CLOCK_MONOTONIC (see libinput:now()).

libinput:timer_cancel()

Cancel the timer for this plugin. This is a no-op if the timer has not been set or has already expired.

libinput:timer_set_absolute(time)

Set a timer for this plugin, with the given time in microseconds. The timeout specifies an absolute time in microseconds (see libinput:now()) The timer will expire once and then call the "timer-expired" event handler (if any).

See libinput:timer_set_relative() for a relative timer.

The following two lines of code are equivalent:

libinput:timer_set_relative(1000000) -- 1 second from now
libinput:timer_set_absolute(libinput:now() + 1000000) -- 1 second from now

Calling this function will cancel any existing (relative or absolute) timer.

libinput:timer_set_relative(timeout)

Set a timer for this plugin, with the given timeout in microseconds from the current time. The timer will expire once and then call the "timer-expired" event handler (if any).

See libinput:timer_set_absolute() for an absolute timer.

The following two lines of code are equivalent:

libinput:timer_set_relative(1000000) -- 1 second from now
libinput:timer_set_absolute(libinput:now() + 1000000) -- 1 second from now

Calling this function will cancel any existing (relative or absolute) timer.

The EvdevDevice type

The EvdevDevice type represents a device available in the system but not (yet) added by libinput. This device may be used to modify a device’s capabilities before the device is processed by libinput.

A plugin should always connect() to the "device-removed" callback to be notified when a device is removed. If the plugin keeps a reference to this device but the device is discarded by libinput, the device’s query methods will return zero values (e.g. nil, 0, an empty table) and methods will be noops.

EvdevDevice:info()

A table containing static information about the device, e.g.

{
   bustype = evdev.BUS_USB,
   vid = 0x1234,
   pid = 0x5678,
}

A plugin must ignore keys it does not know about.

Version 1 of the plugin API supports the following keys and values:

  • bustype: The numeric bustype of the device. See the BUS_* defines in linux/input.h for the list of possible values.

  • vid: The 16-bit vendor ID of the device

  • pid: The 16-bit product ID of the device

If the device has since been discarded by libinput, this function returns an empty table.

EvdevDevice:name()

The device name as set by the kernel

EvdevDevice:usages()

Returns a table of all usages that are currently enabled for this device. Any type that exists on the device has a table assigned and in this table any code that exists on the device is a boolean true. For example:

{
   evdev.REL_X = true,
   evdev.REL_Y = true,
   evdev.BTN_LEFT = true,
}

All other usages are nil, so that the following code is possible:

local usages = device:usages()
if usages[evdev.REL_X] then
   -- do something
end

If the device has since been discarded by libinput, this function returns an empty table.

EvdevDevice:absinfos()

Returns a table of all EV_ABS codes that are currently enabled for this device. The event code is the key, each value is a table containing the following keys: minimum, maximum, fuzz, flat, resolution.

{
   evdev.ABS_X = {
      minimum = 0,
      maximum = 1234,
      fuzz = 0,
      flat = 0,
      resolution = 45,
   },
}

If the device has since been discarded by libinput, this function returns an empty table.

EvdevDevice:udev_properties()

Returns a table containing a filtered list of udev properties available on this device in the form { property_name = property_value, ... }. udev properties used as a boolean (e.g. ID_INPUT) are only present if their value is a logical true.

Version 1 of the plugin API supports the following udev properties:

  • ID_INPUT and all of ID_INPUT_* that denote the device type as assigned by udev. This information is usually used by libinput to determine a device type. Note that for historical reasons these properties have varying rules - some properties may be mutually exclusive, others are independent, others may only be set if another property is set. Refer to the udev documentation (if any) for details. ID_INPUT_WIDTH_MM and ID_INPUT_HEIGHT_MM are excluded from this set.

If the device has since been discarded by libinput, this function returns an empty table.

EvdevDevice:enable_evdev_usage(usage)

Enable the given evdev usage for this device. Use The evdev global for better readability, e.g. device:enable_evdev_usage(evdev.REL_X). This function must not be used for ABS_* events, use set_absinfo() instead.

Once a usage is enabled, events for that usage may be added to a device’s frame.

If the device has since been discarded by libinput, this function does nothing.

EvdevDevice:disable_evdev_usage(usage)

Disable the given evdev usage for this device. Use The evdev global for better readability, e.g. device:disable_evdev_usage(evdev.REL_X).

Once a usage is disabled, events for that usage are discarded from any device frame.

If the device has since been discarded by libinput, this function does nothing.

EvdevDevice:set_absinfo(usage, absinfo)

Set the absolute axis information for the given evdev usage and enable it if it does not yet exist on the device. The absinfo argument is a table containing zero or more of the following keys: minimum, maximum, fuzz, flat, resolution. Any missing key defaults the corresponding value from the device if the device already has this event usage or zero otherwise. For example, the following code changes the resolution but leaves everything else as-is:

local absinfo = {
   resolution = 40,
}
device:set_absinfo(evdev.ABS_X, absinfo)
device:set_absinfo(evdev.ABS_Y, absinfo)

Use The evdev global for better readability as shown in the example above.

If the device has since been discarded by libinput, this function does nothing.

Note

Overriding the absinfo values often indicates buggy firmware. This should typically be fixed with an entry in the 60-evdev.hwdb or Device quirks instead of a plugin so all users of that device can benefit from the fix.

EvdevDevice:connect(name, function)

Set the callback to the given event name. Only one callback may be set for an event name at any time, subsequent callbacks will overwrite any earlier callbacks for the same name.

If the device has since been discarded by libinput, this function does nothing.

Version 1 of the plugin API supports the following events and callback arguments:

  • "evdev-frame": A new evdev frame has started for this device. If the callback returns a value other than nil, that value is the frame with any modified events. An empty frame ({}) causes libinput to drop the current event frame.

    device:connect("evdev-frame", function (device, frame, timestamp)
        -- change any event into a movement left by 1 pixel
        move_left = {
              { usage = evdev.REL_X, value = -1, },
        }
        return move_left
    end
    

    The timestamp of an event frame is in microseconds in CLOCK_MONOTONIC, see libinput:now() for details.

    For performance reasons plugins that do not modify the event frame should return nil (or nothing) instead of the event frame that was passed as argument.

  • "device-removed": This device was removed by libinput. This may happen without the device ever becoming a libinput device as seen by libinput’s public API (e.g. if the device does not meet the requirements to be added). Once this callback is invoked, the plugin should remove any references to this device and stop using it.

    device:connect("device-removed", function (device) ... end)
    

    Functions to query the device’s capabilities (e.g. usages()) will return an empty table.

EvdevDevice:disconnect(name)

Disconnect the existing callback (if any) for the given event name. See EvdevDevice:connect() for a list of supported names.

EvdevDevice:prepend_frame(frame)

Prepend an evdev frame for this device before the current frame (if any). The next plugin will see the prepended frame first followed by the current frame.

This function can only be called from within a device’s "evdev-frame" handler or from within the plugin’s timer callback function.

For example, to change a single event into a drag, prepend a button down and append a button up before each event:

function frame_handler(device, frame, timestamp)
    device:prepend_frame({
        { usage = evdev.BTN_LEFT, value = 1}
    })
    device:append_frame({
        { usage = evdev.BTN_LEFT, value = 0}
    })
    return nil  -- return the current frame unmodified

    -- The next plugin sees the event sequence:
    --    button down, frame, button up
end

If called from within the plugin’s timer there is no current frame and this function is identical to append_frame().

EvdevDevice:append_frame(frame)

Appends an evdev frame for this device after the current frame (if any). This function can only be called from within a device’s "evdev-frame" handler or from within the plugin’s timer callback function.

If called from within the plugin’s timer there is no current frame and this function is identical to prepend_frame().

See prepend_frame() for more details.

EvdevDevice:disable_feature(feature_name)

Disable the given libinput-internal feature for this device. This should be used by plugins that replace that feature with a custom implementation for this device.

libinput may have multiple internal implementations for any given feature, disabling it via this API disables any and all of those implementations, causing the feature to no longer work at all. It is up to the plugin implementation to re-implement that feature to match the user’s expectation.

Version 1 of the plugin API supports the following features: