Synaptics Touchpad

Introduction

The Synaptics Touchpad is a pointing device found on many laptops and supported since FreeBSD 8.

By default, it is recognized and configured as a 3-buttons mouse. Additional features can be enabled with the hw.psm.synaptics_support sysctl tunable.

Also check out FreeBSD FAQ: My laptop has a Synaptics Touchpad. Can I use it in X?

Features

Setup

To enable psm(4) Synaptics support, add the following line to /boot/loader.conf

Enable moused(8). Add the following to /etc/rc.conf

To use it within X.Org, disable the X.Org-specific Synaptics driver, and configure a sysmouse mouse. Here's an example section for /etc/X11/xorg.conf

Horizontal scrolling is disabled by default. To enable it, add the following to /etc/sysctl.conf

Touchpad Configuration

Palm Detection

Finger Pressure

hw.psm.synaptics.min_pressure: 16
hw.psm.synaptics.max_pressure: 220

If the pressure is less than min_pressure or above max_pressure, the packet is ignored but the current action isn't terminated. The pressure is comprised between 0 and 255. Here are some meaningful values, as stated by Synaptics specifications:

Finger Width

hw.psm.synaptics.max_width: 10

If the finger width is above max_width, the packet is ignored but the current action isn't terminated. The width is comprised between 4 and 15. Here are some meningful values, as stated by Synaptics specifications:

Movement Smoothing

For movement smoothing, we use a weighted average then a division where both the weight and the divisor vary, depending on the movement speed. See "How it works".

hw.psm.synaptics.weight_current: 3
hw.psm.synaptics.weight_previous: 6
hw.psm.synaptics.weight_previous_na: 20
hw.psm.synaptics.weight_len_squared: 2000

weight_current and weight_previous define the range of weights to use when calculating the average. weight_previous_na is used instead of weight_previous when the finger is inside the noisy area (see "Touchpad borders").

TODO: explain weight_len_squared.

hw.psm.synaptics.div_min: 9
hw.psm.synaptics.div_max: 17
hw.psm.synaptics.div_max_na: 30
hw.psm.synaptics.div_len: 100

div_min and div_max define the range of divisors to when calculating the final delta for x & y. div_max_na is used instead of div_max when the finger is inside the noisy area (see "Touchpad borders").

TODO: explain div_len.

hw.psm.synaptics.multiplicator: 10000

This multiplicator is used during averages and divisions to increase the precision.

How Movement Smoothing Works

The smoothing is based on a weighted average:

   1 average = (weight_current * delta + weight_previous * average) / (weight_current + weight_preivous)

where :

This average is calculated for each axis (x & y) separately.

The problem with a simple weighted average as above is that the user has to choose between a smooth but laggy movement (weight_previous greater then weight_current) or a sharp but imprecise movement at slow speed (weight_previous smaller or equal to weight_current).

The idea is then to change weight_previous based on the last packets:

Internaly, the driver keeps the last 10 packets. Here is the simplified algorithm:

   1 dx = abs(new_x - older_x) + 1; /* +1: to avoid a division by zero below. */
   2 dy = abs(new_y - older_y) + 1;
   3 
   4 /* "len" is really the length squared. */
   5 len = (dx * dx) + (dy * dy);
   6 
   7 /* "sysctl_weight_len_squared" is the length squared at which the
   8  * "weight_previous" will start to decrease. */
   9 weight_previous = sysctl_weight_len_squared * weight_previous / len;
  10 weight_previous = imin(weight_previous, sysctl_weight_previous);

The driver will do almost the same with the divisor to calculate the acceleration. The rest of the division is also kept between each packet; otherwise, with slow movements, the pointer doesn't reproduce well what the finger is doing.

Touchpad Borders (Margins)

Touchpad edges (borders) can generate a lot of noise in reported coordinates. There are two sets of sysctl tunables that control this area:

hw.psm.synaptics.margin_top: 200
hw.psm.synaptics.margin_right: 200
hw.psm.synaptics.margin_bottom: 200
hw.psm.synaptics.margin_left: 200

These margins act as a high-pass filter: for instance, the point [50; 170] will be reported as [200;200].

hw.psm.synaptics.na_top: 1783
hw.psm.synaptics.na_right: 563
hw.psm.synaptics.na_bottom: 1408
hw.psm.synaptics.na_left: 1600

na stands for noisy area. Movements inside these larger margins will be smoothed using different max weight and divisor. See "Movement Smoothing".

Points History

hw.psm.synaptics.window_min: 4
hw.psm.synaptics.window_max: 10

window_min indicates the minimum number of packets to receive before an action is considered to be real and wanted. window_max is the maximum number of packets to use in calculations.

Tap and Tap-hold

hw.psm.synaptics.tap_max_delta: 80
hw.psm.synaptics.tap_min_queue: 2

tap_max_delta is the maximum delta between points to treat the action as a tap. This is to prevent any unwanted click. tap_min_queue is the minimum number of packets needed to consider a tap.

hw.psm.synaptics.taphold_timeout: 125000

taphold_timeout is the time (in microseconds) between two taps to consider a tap-hols action.

Virtual Scrolling

hw.psm.synaptics.vscroll_hor_area: 1300
hw.psm.synaptics.vscroll_ver_area: -600
hw.psm.synaptics.vscroll_min_delta: 50
hw.psm.synaptics.vscroll_div_min: 100
hw.psm.synaptics.vscroll_div_max: 150

An area must be dedicated to virtual scrolling for now. This area is defined by vscroll_hor_area (positive for an area at the bottom, negative for an area at the top) for horizontal scrolling and vscroll_ver_area (positive for an area on the left, negative for an area on the right) for vertical scrolling. vscroll_min_delta is the minimum delta between points to consider a scrolling action. This is to prevent unwanted scrolling when tapping for instance. vscroll_div_min and vscroll_div_max are the divisors used instead of div_min and div_max respectively.

Toggling Synaptics Touchpad with devd

Although touchpads can support palm detection, it is sometimes desirable to disable them if a USB mouse is plugged in.

These instructions assume a USB wired or wireless mouse. For a mouse that connects via Bluetooth, you will also need to identify your mouse device-name.

Setup

We will be triggering a script from devd in order to toggle synaptics.

Create a new script in /usr/local/bin named "synaptics-hotplug".

In this script, paste the following:

trackpad_disable(){
        /sbin/sysctl hw.psm.synaptics.min_pressure=220
}

trackpad_enable(){
        /sbin/kldunload ums
        /sbin/sysctl hw.psm.synaptics.min_pressure=32
}

synaptics_$1

Then you will want to make the script executable.

chmod +x synaptics-hotplug

Next you will want to open /etc/devd/usb.conf and navigate to line 5686, it should look like the following:

nomatch 32 {
        match "bus" "uhub[0-9]+";
        match "mode" "host";
        match "intclass" "0x03";
        match "intsubclass" "0x01";
        match "intprotocol" "0x02";
        action "kldload -n ums";
};

That last line in the block, 'action' is what runs everytime we plug in our USB mouse, edit it so it runs our script:

action "kldload -n ums && /usr/local/bin/synaptics-hotplug disable";

Now everytime we plug in our USB mouse, it will run our script with the "disable" argument variable which runs our script's "synaptics_disable" function, this changes our synaptics trackpad minimal pressure to 220, disabling it.

Next open /etc/devd.conf as root and put the following at the end of the file(or anywhere):

detach 100 {
        device-name "ums[0-9]+";
        action "/usr/local/bin/synaptics-hotplug enable";
};

With this done, the only thing you'll need to do is restart devd

service devd restart

Once devd is restarted, the synaptics trackpad should turn off and on depending whether your USB mouse is plugged in.

Credit for scripted synpatics toggle via devd goes to SamuelGogarty.

Other Resources


CategoryHowTo CategoryLaptop

SynapticsTouchpad (last edited 2022-08-18T05:58:22+0000 by KubilayKocak)