Compare commits
16 commits
master
...
feature/sd
Author | SHA1 | Date | |
---|---|---|---|
|
64bbdc81a2 | ||
|
d6dc3403d8 | ||
|
4f100956ac | ||
|
ac5736a90f | ||
|
bf6df9e889 | ||
|
a5a4415003 | ||
|
40c9669e78 | ||
|
8767796562 | ||
|
aa1981e280 | ||
|
9d12a4edb6 | ||
|
d81e46b7a0 | ||
|
ffd30bd3d0 | ||
|
2d7f2c67bf | ||
|
4a8ab83a13 | ||
|
6c291aeb7a | ||
|
13d2a6a69b |
76 changed files with 2951 additions and 5988 deletions
|
@ -1,6 +1,6 @@
|
||||||
include:
|
include:
|
||||||
- local: '.gitlab-ci/libfprint-templates.yaml'
|
- local: '.gitlab-ci/libfprint-templates.yaml'
|
||||||
- project: 'freedesktop/ci-templates'
|
- project: 'wayland/ci-templates'
|
||||||
ref: master
|
ref: master
|
||||||
file: '/templates/fedora.yml'
|
file: '/templates/fedora.yml'
|
||||||
- remote: 'https://gitlab.gnome.org/GNOME/citemplates/-/raw/master/flatpak/flatpak_ci_initiative.yml'
|
- remote: 'https://gitlab.gnome.org/GNOME/citemplates/-/raw/master/flatpak/flatpak_ci_initiative.yml'
|
||||||
|
@ -9,7 +9,6 @@ variables:
|
||||||
extends: .libfprint_common_variables
|
extends: .libfprint_common_variables
|
||||||
FDO_DISTRIBUTION_TAG: latest
|
FDO_DISTRIBUTION_TAG: latest
|
||||||
FDO_DISTRIBUTION_VERSION: rawhide
|
FDO_DISTRIBUTION_VERSION: rawhide
|
||||||
FDO_UPSTREAM_REPO: "libfprint/$CI_PROJECT_NAME"
|
|
||||||
FEDORA_IMAGE: "$CI_REGISTRY/libfprint/$CI_PROJECT_NAME/fedora/$FDO_DISTRIBUTION_VERSION:$FDO_DISTRIBUTION_TAG"
|
FEDORA_IMAGE: "$CI_REGISTRY/libfprint/$CI_PROJECT_NAME/fedora/$FDO_DISTRIBUTION_VERSION:$FDO_DISTRIBUTION_TAG"
|
||||||
BUNDLE: "org.freedesktop.libfprint.Demo.flatpak"
|
BUNDLE: "org.freedesktop.libfprint.Demo.flatpak"
|
||||||
LAST_ABI_BREAK: "056ea541ddc97f5806cffbd99a12dc87e4da3546"
|
LAST_ABI_BREAK: "056ea541ddc97f5806cffbd99a12dc87e4da3546"
|
||||||
|
@ -20,7 +19,7 @@ stages:
|
||||||
- test
|
- test
|
||||||
- flatpak
|
- flatpak
|
||||||
|
|
||||||
image: $FEDORA_IMAGE
|
image: "$FEDORA_IMAGE"
|
||||||
|
|
||||||
.build_one_driver_template: &build_one_driver
|
.build_one_driver_template: &build_one_driver
|
||||||
script:
|
script:
|
||||||
|
@ -63,14 +62,13 @@ test:
|
||||||
variables:
|
variables:
|
||||||
- $CI_PIPELINE_SOURCE == "schedule"
|
- $CI_PIPELINE_SOURCE == "schedule"
|
||||||
script:
|
script:
|
||||||
- meson --werror -Ddrivers=all -Db_coverage=true . _build
|
- meson --werror -Ddrivers=all -Db_coverage=true -Dsdcp_virt_binary=/usr/local/bin/sdcp_virt_device . _build
|
||||||
- ninja -C _build
|
- ninja -C _build
|
||||||
- meson test -C _build --print-errorlogs --no-stdsplit --timeout-multiplier 3
|
- meson test -C _build --print-errorlogs --no-stdsplit --timeout-multiplier 3
|
||||||
- ninja -C _build coverage
|
- ninja -C _build coverage
|
||||||
- cat _build/meson-logs/coverage.txt
|
- cat _build/meson-logs/coverage.txt
|
||||||
artifacts:
|
artifacts:
|
||||||
expose_as: 'Coverage Report'
|
expose_as: 'Coverage Report'
|
||||||
when: always
|
|
||||||
paths:
|
paths:
|
||||||
- _build/meson-logs
|
- _build/meson-logs
|
||||||
- _build/meson-logs/coveragereport/index.html
|
- _build/meson-logs/coveragereport/index.html
|
||||||
|
@ -83,12 +81,11 @@ test_valgrind:
|
||||||
variables:
|
variables:
|
||||||
- $CI_PIPELINE_SOURCE == "schedule"
|
- $CI_PIPELINE_SOURCE == "schedule"
|
||||||
script:
|
script:
|
||||||
- meson -Ddrivers=all . _build
|
- meson -Ddrivers=all -Dsdcp_virt_binary=/usr/local/bin/sdcp_virt_device . _build
|
||||||
- ninja -C _build
|
- ninja -C _build
|
||||||
- meson test -C _build --print-errorlogs --no-stdsplit --setup=valgrind
|
- meson test -C _build --print-errorlogs --no-stdsplit --setup=valgrind
|
||||||
artifacts:
|
artifacts:
|
||||||
expose_as: 'Valgrind test logs'
|
expose_as: 'Valgrind test logs'
|
||||||
when: always
|
|
||||||
paths:
|
paths:
|
||||||
- _build/meson-logs
|
- _build/meson-logs
|
||||||
- _build/meson-logs/testlog-valgrind.txt
|
- _build/meson-logs/testlog-valgrind.txt
|
||||||
|
@ -160,15 +157,26 @@ container_fedora_build:
|
||||||
- $CI_PIPELINE_SOURCE == "schedule" && $CRON_TASK == "BUILD_CI_IMAGES"
|
- $CI_PIPELINE_SOURCE == "schedule" && $CRON_TASK == "BUILD_CI_IMAGES"
|
||||||
variables:
|
variables:
|
||||||
GIT_STRATEGY: none # no need to pull the whole tree for rebuilding the image
|
GIT_STRATEGY: none # no need to pull the whole tree for rebuilding the image
|
||||||
FDO_FORCE_REBUILD: 1
|
|
||||||
# a list of packages to install
|
# a list of packages to install
|
||||||
FDO_DISTRIBUTION_PACKAGES:
|
FDO_DISTRIBUTION_PACKAGES:
|
||||||
$LIBFPRINT_DEPENDENCIES
|
$LIBFPRINT_DEPENDENCIES
|
||||||
vala
|
vala
|
||||||
libpcap-devel
|
libpcap-devel
|
||||||
libudev-devel
|
libudev-devel
|
||||||
|
mbedtls-devel
|
||||||
FDO_DISTRIBUTION_EXEC: |
|
FDO_DISTRIBUTION_EXEC: |
|
||||||
git clone https://github.com/martinpitt/umockdev.git && \
|
git clone https://github.com/martinpitt/umockdev.git && \
|
||||||
cd umockdev && \
|
cd umockdev && \
|
||||||
meson _build --prefix=/usr && \
|
meson _build --prefix=/usr && \
|
||||||
ninja -C _build && ninja -C _build install
|
ninja -C _build && ninja -C _build install
|
||||||
|
cd /tmp
|
||||||
|
mkdir -p /usr/local/bin
|
||||||
|
git clone https://github.com/benzea/SecureDeviceConnectionProtocol.git
|
||||||
|
# Don't bother with cmake
|
||||||
|
gcc -l mbedcrypto -I SecureDeviceConnectionProtocol/src/include \
|
||||||
|
SecureDeviceConnectionProtocol/src/test/virt_device.c \
|
||||||
|
SecureDeviceConnectionProtocol/src/client/client.c \
|
||||||
|
SecureDeviceConnectionProtocol/src/test/helpers.c \
|
||||||
|
SecureDeviceConnectionProtocol/src/test/testkeys.c \
|
||||||
|
-o /usr/local/bin/sdcp_virt_device
|
||||||
|
rm -rf SecureDeviceConnectionProtocol
|
||||||
|
|
|
@ -26,4 +26,3 @@
|
||||||
uncrustify
|
uncrustify
|
||||||
valgrind
|
valgrind
|
||||||
clang-analyzer
|
clang-analyzer
|
||||||
diffutils
|
|
||||||
|
|
21
NEWS
21
NEWS
|
@ -1,27 +1,6 @@
|
||||||
This file lists notable changes in each release. For the full history of all
|
This file lists notable changes in each release. For the full history of all
|
||||||
changes, see ChangeLog.
|
changes, see ChangeLog.
|
||||||
|
|
||||||
2021-06-30: v1.94.0 release
|
|
||||||
|
|
||||||
Highlights:
|
|
||||||
* Implement suspend/resume handling including USB wakeup configuration.
|
|
||||||
This requires writing the "persist" and "wakeup" sysfs attributes.
|
|
||||||
* Add simple temperature module to prevent devices from becoming too hot
|
|
||||||
* Add feature for continuous scanning
|
|
||||||
* New internal "critical section" API to simplify driver development
|
|
||||||
* elan: new PID 0x0c58
|
|
||||||
* elanmoc: Fixes for multi-user handling and FW changes
|
|
||||||
* virtual-device: Do not time out for SCAN command
|
|
||||||
|
|
||||||
2021-06-30: v1.92.1 release
|
|
||||||
|
|
||||||
Highlights:
|
|
||||||
* elanmoc: New driver for ELAN match-on-chip devices
|
|
||||||
* egis0570: New driver for some Egis Technology devices
|
|
||||||
* synaptics: Fix empty identify causing enroll issues
|
|
||||||
* elan: Support more PIDs
|
|
||||||
* misc: Architecture related bugfixes
|
|
||||||
|
|
||||||
2021-06-30: v1.92.0 release
|
2021-06-30: v1.92.0 release
|
||||||
|
|
||||||
Highlights:
|
Highlights:
|
||||||
|
|
|
@ -4,7 +4,6 @@
|
||||||
# Supported by libfprint driver aes1610
|
# Supported by libfprint driver aes1610
|
||||||
usb:v08FFp1600*
|
usb:v08FFp1600*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver aes1660
|
# Supported by libfprint driver aes1660
|
||||||
usb:v08FFp1660*
|
usb:v08FFp1660*
|
||||||
|
@ -25,19 +24,16 @@ usb:v08FFp168D*
|
||||||
usb:v08FFp168E*
|
usb:v08FFp168E*
|
||||||
usb:v08FFp168F*
|
usb:v08FFp168F*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver aes2501
|
# Supported by libfprint driver aes2501
|
||||||
usb:v08FFp2500*
|
usb:v08FFp2500*
|
||||||
usb:v08FFp2580*
|
usb:v08FFp2580*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver aes2550
|
# Supported by libfprint driver aes2550
|
||||||
usb:v08FFp2550*
|
usb:v08FFp2550*
|
||||||
usb:v08FFp2810*
|
usb:v08FFp2810*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver aes2660
|
# Supported by libfprint driver aes2660
|
||||||
usb:v08FFp2660*
|
usb:v08FFp2660*
|
||||||
|
@ -59,23 +55,14 @@ usb:v08FFp268E*
|
||||||
usb:v08FFp268F*
|
usb:v08FFp268F*
|
||||||
usb:v08FFp2691*
|
usb:v08FFp2691*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver aes3500
|
# Supported by libfprint driver aes3500
|
||||||
usb:v08FFp5731*
|
usb:v08FFp5731*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver aes4000
|
# Supported by libfprint driver aes4000
|
||||||
usb:v5501p08FF*
|
usb:v5501p08FF*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver egis0570
|
|
||||||
usb:v1C7Ap0570*
|
|
||||||
usb:v1C7Ap0571*
|
|
||||||
ID_AUTOSUSPEND=1
|
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver elan
|
# Supported by libfprint driver elan
|
||||||
usb:v04F3p0903*
|
usb:v04F3p0903*
|
||||||
|
@ -136,20 +123,11 @@ usb:v04F3p0C42*
|
||||||
usb:v04F3p0C4D*
|
usb:v04F3p0C4D*
|
||||||
usb:v04F3p0C4F*
|
usb:v04F3p0C4F*
|
||||||
usb:v04F3p0C63*
|
usb:v04F3p0C63*
|
||||||
usb:v04F3p0C6E*
|
|
||||||
usb:v04F3p0C58*
|
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver elanmoc
|
|
||||||
usb:v04F3p0C7E*
|
|
||||||
ID_AUTOSUSPEND=1
|
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver etes603
|
# Supported by libfprint driver etes603
|
||||||
usb:v1C7Ap0603*
|
usb:v1C7Ap0603*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver goodixmoc
|
# Supported by libfprint driver goodixmoc
|
||||||
usb:v27C6p5840*
|
usb:v27C6p5840*
|
||||||
|
@ -157,7 +135,6 @@ usb:v27C6p609C*
|
||||||
usb:v27C6p60A2*
|
usb:v27C6p60A2*
|
||||||
usb:v27C6p639C*
|
usb:v27C6p639C*
|
||||||
usb:v27C6p63AC*
|
usb:v27C6p63AC*
|
||||||
usb:v27C6p63BC*
|
|
||||||
usb:v27C6p6496*
|
usb:v27C6p6496*
|
||||||
usb:v27C6p6584*
|
usb:v27C6p6584*
|
||||||
usb:v27C6p658C*
|
usb:v27C6p658C*
|
||||||
|
@ -166,12 +143,10 @@ usb:v27C6p6594*
|
||||||
usb:v27C6p659C*
|
usb:v27C6p659C*
|
||||||
usb:v27C6p6A94*
|
usb:v27C6p6A94*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver nb1010
|
# Supported by libfprint driver nb1010
|
||||||
usb:v298Dp1010*
|
usb:v298Dp1010*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver synaptics
|
# Supported by libfprint driver synaptics
|
||||||
usb:v06CBp00BD*
|
usb:v06CBp00BD*
|
||||||
|
@ -183,29 +158,22 @@ usb:v06CBp00C9*
|
||||||
usb:v06CBp0100*
|
usb:v06CBp0100*
|
||||||
usb:v06CBp00F0*
|
usb:v06CBp00F0*
|
||||||
usb:v06CBp0103*
|
usb:v06CBp0103*
|
||||||
usb:v06CBp0123*
|
|
||||||
usb:v06CBp0126*
|
|
||||||
usb:v06CBp0129*
|
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver upeksonly
|
# Supported by libfprint driver upeksonly
|
||||||
usb:v147Ep2016*
|
usb:v147Ep2016*
|
||||||
usb:v147Ep1000*
|
usb:v147Ep1000*
|
||||||
usb:v147Ep1001*
|
usb:v147Ep1001*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver upektc
|
# Supported by libfprint driver upektc
|
||||||
usb:v0483p2015*
|
usb:v0483p2015*
|
||||||
usb:v147Ep3001*
|
usb:v147Ep3001*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver upektc_img
|
# Supported by libfprint driver upektc_img
|
||||||
usb:v147Ep2020*
|
usb:v147Ep2020*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver uru4000
|
# Supported by libfprint driver uru4000
|
||||||
usb:v045Ep00BC*
|
usb:v045Ep00BC*
|
||||||
|
@ -215,28 +183,23 @@ usb:v05BAp0007*
|
||||||
usb:v05BAp0008*
|
usb:v05BAp0008*
|
||||||
usb:v05BAp000A*
|
usb:v05BAp000A*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver vcom5s
|
# Supported by libfprint driver vcom5s
|
||||||
usb:v061Ap0110*
|
usb:v061Ap0110*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver vfs0050
|
# Supported by libfprint driver vfs0050
|
||||||
usb:v138Ap0050*
|
usb:v138Ap0050*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver vfs101
|
# Supported by libfprint driver vfs101
|
||||||
usb:v138Ap0001*
|
usb:v138Ap0001*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver vfs301
|
# Supported by libfprint driver vfs301
|
||||||
usb:v138Ap0005*
|
usb:v138Ap0005*
|
||||||
usb:v138Ap0008*
|
usb:v138Ap0008*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver vfs5011
|
# Supported by libfprint driver vfs5011
|
||||||
usb:v138Ap0010*
|
usb:v138Ap0010*
|
||||||
|
@ -245,12 +208,10 @@ usb:v138Ap0015*
|
||||||
usb:v138Ap0017*
|
usb:v138Ap0017*
|
||||||
usb:v138Ap0018*
|
usb:v138Ap0018*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Supported by libfprint driver vfs7552
|
# Supported by libfprint driver vfs7552
|
||||||
usb:v138Ap0091*
|
usb:v138Ap0091*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
||||||
# Known unsupported devices
|
# Known unsupported devices
|
||||||
usb:v04F3p036B*
|
usb:v04F3p036B*
|
||||||
|
@ -273,7 +234,6 @@ usb:v06CBp00C4*
|
||||||
usb:v06CBp00CB*
|
usb:v06CBp00CB*
|
||||||
usb:v06CBp00D8*
|
usb:v06CBp00D8*
|
||||||
usb:v06CBp00DA*
|
usb:v06CBp00DA*
|
||||||
usb:v06CBp00E7*
|
|
||||||
usb:v06CBp00E9*
|
usb:v06CBp00E9*
|
||||||
usb:v0A5Cp5801*
|
usb:v0A5Cp5801*
|
||||||
usb:v0A5Cp5805*
|
usb:v0A5Cp5805*
|
||||||
|
@ -282,7 +242,6 @@ usb:v0A5Cp5840*
|
||||||
usb:v0A5Cp5841*
|
usb:v0A5Cp5841*
|
||||||
usb:v0A5Cp5842*
|
usb:v0A5Cp5842*
|
||||||
usb:v0A5Cp5843*
|
usb:v0A5Cp5843*
|
||||||
usb:v0A5Cp5844*
|
|
||||||
usb:v0A5Cp5845*
|
usb:v0A5Cp5845*
|
||||||
usb:v10A5p0007*
|
usb:v10A5p0007*
|
||||||
usb:v1188p9545*
|
usb:v1188p9545*
|
||||||
|
@ -301,6 +260,7 @@ usb:v147Ep1002*
|
||||||
usb:v1491p0088*
|
usb:v1491p0088*
|
||||||
usb:v16D1p1027*
|
usb:v16D1p1027*
|
||||||
usb:v1C7Ap0300*
|
usb:v1C7Ap0300*
|
||||||
|
usb:v1C7Ap0570*
|
||||||
usb:v1C7Ap0575*
|
usb:v1C7Ap0575*
|
||||||
usb:v27C6p5042*
|
usb:v27C6p5042*
|
||||||
usb:v27C6p5110*
|
usb:v27C6p5110*
|
||||||
|
@ -325,4 +285,3 @@ usb:v2808p9338*
|
||||||
usb:v298Dp2033*
|
usb:v298Dp2033*
|
||||||
usb:v3538p0930*
|
usb:v3538p0930*
|
||||||
ID_AUTOSUSPEND=1
|
ID_AUTOSUSPEND=1
|
||||||
ID_PERSIST=0
|
|
||||||
|
|
|
@ -50,8 +50,6 @@ fp_device_capture
|
||||||
fp_device_delete_print
|
fp_device_delete_print
|
||||||
fp_device_list_prints
|
fp_device_list_prints
|
||||||
fp_device_clear_storage
|
fp_device_clear_storage
|
||||||
fp_device_suspend
|
|
||||||
fp_device_resume
|
|
||||||
fp_device_open_finish
|
fp_device_open_finish
|
||||||
fp_device_close_finish
|
fp_device_close_finish
|
||||||
fp_device_enroll_finish
|
fp_device_enroll_finish
|
||||||
|
@ -61,8 +59,6 @@ fp_device_capture_finish
|
||||||
fp_device_delete_print_finish
|
fp_device_delete_print_finish
|
||||||
fp_device_list_prints_finish
|
fp_device_list_prints_finish
|
||||||
fp_device_clear_storage_finish
|
fp_device_clear_storage_finish
|
||||||
fp_device_suspend_finish
|
|
||||||
fp_device_resume_finish
|
|
||||||
fp_device_open_sync
|
fp_device_open_sync
|
||||||
fp_device_close_sync
|
fp_device_close_sync
|
||||||
fp_device_enroll_sync
|
fp_device_enroll_sync
|
||||||
|
@ -72,8 +68,6 @@ fp_device_capture_sync
|
||||||
fp_device_delete_print_sync
|
fp_device_delete_print_sync
|
||||||
fp_device_list_prints_sync
|
fp_device_list_prints_sync
|
||||||
fp_device_clear_storage_sync
|
fp_device_clear_storage_sync
|
||||||
fp_device_suspend_sync
|
|
||||||
fp_device_resume_sync
|
|
||||||
FpDevice
|
FpDevice
|
||||||
</SECTION>
|
</SECTION>
|
||||||
|
|
||||||
|
@ -100,6 +94,12 @@ FP_TYPE_IMAGE_DEVICE
|
||||||
FpImageDevice
|
FpImageDevice
|
||||||
</SECTION>
|
</SECTION>
|
||||||
|
|
||||||
|
<SECTION>
|
||||||
|
<FILE>fp-sdcp-device</FILE>
|
||||||
|
FP_TYPE_SDCP_DEVICE
|
||||||
|
FpSdcpDevice
|
||||||
|
</SECTION>
|
||||||
|
|
||||||
<SECTION>
|
<SECTION>
|
||||||
<FILE>fp-print</FILE>
|
<FILE>fp-print</FILE>
|
||||||
FP_TYPE_PRINT
|
FP_TYPE_PRINT
|
||||||
|
@ -166,8 +166,6 @@ fpi_device_add_timeout
|
||||||
fpi_device_set_nr_enroll_stages
|
fpi_device_set_nr_enroll_stages
|
||||||
fpi_device_set_scan_type
|
fpi_device_set_scan_type
|
||||||
fpi_device_update_features
|
fpi_device_update_features
|
||||||
fpi_device_critical_enter
|
|
||||||
fpi_device_critical_leave
|
|
||||||
fpi_device_remove
|
fpi_device_remove
|
||||||
fpi_device_report_finger_status
|
fpi_device_report_finger_status
|
||||||
fpi_device_report_finger_status_changes
|
fpi_device_report_finger_status_changes
|
||||||
|
@ -180,12 +178,10 @@ fpi_device_verify_complete
|
||||||
fpi_device_identify_complete
|
fpi_device_identify_complete
|
||||||
fpi_device_capture_complete
|
fpi_device_capture_complete
|
||||||
fpi_device_delete_complete
|
fpi_device_delete_complete
|
||||||
fpi_device_list_complete
|
|
||||||
fpi_device_suspend_complete
|
|
||||||
fpi_device_resume_complete
|
|
||||||
fpi_device_enroll_progress
|
fpi_device_enroll_progress
|
||||||
fpi_device_verify_report
|
fpi_device_verify_report
|
||||||
fpi_device_identify_report
|
fpi_device_identify_report
|
||||||
|
fpi_device_list_complete
|
||||||
fpi_device_class_auto_initialize_features
|
fpi_device_class_auto_initialize_features
|
||||||
</SECTION>
|
</SECTION>
|
||||||
|
|
||||||
|
@ -214,6 +210,26 @@ fpi_image_device_retry_scan
|
||||||
fpi_image_device_set_bz3_threshold
|
fpi_image_device_set_bz3_threshold
|
||||||
</SECTION>
|
</SECTION>
|
||||||
|
|
||||||
|
<SECTION>
|
||||||
|
<FILE>fpi-sdcp-device</FILE>
|
||||||
|
<TITLE>Internal FpSdcpDevice</TITLE>
|
||||||
|
FpiSdcpClaim
|
||||||
|
FpSdcpDeviceClass
|
||||||
|
fpi_sdcp_claim_copy
|
||||||
|
fpi_sdcp_claim_free
|
||||||
|
fpi_sdcp_claim_get_type
|
||||||
|
fpi_sdcp_claim_new
|
||||||
|
fpi_sdcp_device_connect_complete
|
||||||
|
fpi_sdcp_device_get_connect_data
|
||||||
|
fpi_sdcp_device_get_reconnect_data
|
||||||
|
fpi_sdcp_device_reconnect_complete
|
||||||
|
fpi_sdcp_device_enroll_commit_complete
|
||||||
|
fpi_sdcp_device_enroll_ready
|
||||||
|
fpi_sdcp_device_enroll_set_nonce
|
||||||
|
fpi_sdcp_device_identify_retry
|
||||||
|
fpi_sdcp_device_identify_complete
|
||||||
|
</SECTION>
|
||||||
|
|
||||||
<SECTION>
|
<SECTION>
|
||||||
<FILE>fpi-log</FILE>
|
<FILE>fpi-log</FILE>
|
||||||
fp_dbg
|
fp_dbg
|
||||||
|
|
|
@ -28,6 +28,7 @@
|
||||||
<xi:include href="xml/fp-context.xml"/>
|
<xi:include href="xml/fp-context.xml"/>
|
||||||
<xi:include href="xml/fp-device.xml"/>
|
<xi:include href="xml/fp-device.xml"/>
|
||||||
<xi:include href="xml/fp-image-device.xml"/>
|
<xi:include href="xml/fp-image-device.xml"/>
|
||||||
|
<xi:include href="xml/fp-sdcp-device.xml"/>
|
||||||
<xi:include href="xml/fp-print.xml"/>
|
<xi:include href="xml/fp-print.xml"/>
|
||||||
<xi:include href="xml/fp-image.xml"/>
|
<xi:include href="xml/fp-image.xml"/>
|
||||||
</part>
|
</part>
|
||||||
|
@ -38,6 +39,7 @@
|
||||||
<title>Device methods for drivers</title>
|
<title>Device methods for drivers</title>
|
||||||
<xi:include href="xml/fpi-device.xml"/>
|
<xi:include href="xml/fpi-device.xml"/>
|
||||||
<xi:include href="xml/fpi-image-device.xml"/>
|
<xi:include href="xml/fpi-image-device.xml"/>
|
||||||
|
<xi:include href="xml/fpi-sdcp-device.xml"/>
|
||||||
</chapter>
|
</chapter>
|
||||||
|
|
||||||
<chapter id="driver-helpers">
|
<chapter id="driver-helpers">
|
||||||
|
|
|
@ -365,7 +365,7 @@ capture_read_stripe_data_cb (FpiUsbTransfer *transfer,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
fp_dbg ("Got %" G_GSIZE_FORMAT " bytes of data", actual_length);
|
fp_dbg ("Got %lu bytes of data", actual_length);
|
||||||
while (actual_length)
|
while (actual_length)
|
||||||
{
|
{
|
||||||
gssize payload_length;
|
gssize payload_length;
|
||||||
|
@ -386,7 +386,7 @@ capture_read_stripe_data_cb (FpiUsbTransfer *transfer,
|
||||||
(priv->stripe_packet->data[AESX660_RESPONSE_SIZE_MSB_OFFSET] << 8);
|
(priv->stripe_packet->data[AESX660_RESPONSE_SIZE_MSB_OFFSET] << 8);
|
||||||
fp_dbg ("Got frame, type %.2x payload of size %.4lx",
|
fp_dbg ("Got frame, type %.2x payload of size %.4lx",
|
||||||
priv->stripe_packet->data[AESX660_RESPONSE_TYPE_OFFSET],
|
priv->stripe_packet->data[AESX660_RESPONSE_TYPE_OFFSET],
|
||||||
(long) payload_length);
|
payload_length);
|
||||||
|
|
||||||
still_needed_len = MAX (0, AESX660_HEADER_SIZE + payload_length - (gssize) priv->stripe_packet->len);
|
still_needed_len = MAX (0, AESX660_HEADER_SIZE + payload_length - (gssize) priv->stripe_packet->len);
|
||||||
copy_len = MIN (actual_length, still_needed_len);
|
copy_len = MIN (actual_length, still_needed_len);
|
||||||
|
@ -441,7 +441,7 @@ capture_run_state (FpiSsm *ssm, FpDevice *_dev)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case CAPTURE_SET_IDLE:
|
case CAPTURE_SET_IDLE:
|
||||||
fp_dbg ("Got %" G_GSIZE_FORMAT " frames", priv->strips_len);
|
fp_dbg ("Got %lu frames", priv->strips_len);
|
||||||
aesX660_send_cmd (ssm, _dev, set_idle_cmd, sizeof (set_idle_cmd),
|
aesX660_send_cmd (ssm, _dev, set_idle_cmd, sizeof (set_idle_cmd),
|
||||||
capture_set_idle_cmd_cb);
|
capture_set_idle_cmd_cb);
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -1,444 +0,0 @@
|
||||||
/*
|
|
||||||
* Egis Technology Inc. (aka. LighTuning) 0570 driver for libfprint
|
|
||||||
* Copyright (C) 2021 Maxim Kolesnikov <kolesnikov@svyazcom.ru>
|
|
||||||
* Copyright (C) 2021 Saeed/Ali Rk <saeed.ali.rahimi@gmail.com>
|
|
||||||
*
|
|
||||||
* This library is free software; you can redistribute it and/or
|
|
||||||
* modify it under the terms of the GNU Lesser General Public
|
|
||||||
* License as published by the Free Software Foundation; either
|
|
||||||
* version 2.1 of the License, or (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This library is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
||||||
* Lesser General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Lesser General Public
|
|
||||||
* License along with this library; if not, write to the Free Software
|
|
||||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
||||||
*/
|
|
||||||
|
|
||||||
#define FP_COMPONENT "egis0570"
|
|
||||||
|
|
||||||
#include "egis0570.h"
|
|
||||||
#include "drivers_api.h"
|
|
||||||
|
|
||||||
/* Packet types */
|
|
||||||
#define PKT_TYPE_INIT 0
|
|
||||||
#define PKT_TYPE_REPEAT 1
|
|
||||||
|
|
||||||
/* Struct */
|
|
||||||
struct _FpDeviceEgis0570
|
|
||||||
{
|
|
||||||
FpImageDevice parent;
|
|
||||||
|
|
||||||
gboolean running;
|
|
||||||
gboolean stop;
|
|
||||||
|
|
||||||
GSList *strips;
|
|
||||||
guint8 *background;
|
|
||||||
gsize strips_len;
|
|
||||||
|
|
||||||
int pkt_num;
|
|
||||||
int pkt_type;
|
|
||||||
};
|
|
||||||
G_DECLARE_FINAL_TYPE (FpDeviceEgis0570, fpi_device_egis0570, FPI, DEVICE_EGIS0570, FpImageDevice);
|
|
||||||
G_DEFINE_TYPE (FpDeviceEgis0570, fpi_device_egis0570, FP_TYPE_IMAGE_DEVICE);
|
|
||||||
|
|
||||||
static unsigned char
|
|
||||||
egis_get_pixel (struct fpi_frame_asmbl_ctx *ctx, struct fpi_frame *frame, unsigned int x, unsigned int y)
|
|
||||||
{
|
|
||||||
return frame->data[x + y * ctx->frame_width];
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct fpi_frame_asmbl_ctx assembling_ctx = {
|
|
||||||
.frame_width = EGIS0570_IMGWIDTH,
|
|
||||||
.frame_height = EGIS0570_RFMGHEIGHT,
|
|
||||||
.image_width = EGIS0570_IMGWIDTH * 4 / 3,
|
|
||||||
.get_pixel = egis_get_pixel,
|
|
||||||
};
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Service
|
|
||||||
*/
|
|
||||||
|
|
||||||
static gboolean
|
|
||||||
is_last_pkt (FpDevice *dev)
|
|
||||||
{
|
|
||||||
FpDeviceEgis0570 *self = FPI_DEVICE_EGIS0570 (dev);
|
|
||||||
|
|
||||||
int type = self->pkt_type;
|
|
||||||
int num = self->pkt_num;
|
|
||||||
|
|
||||||
gboolean r;
|
|
||||||
|
|
||||||
r = ((type == PKT_TYPE_INIT) && (num == (EGIS0570_INIT_TOTAL - 1)));
|
|
||||||
r |= ((type == PKT_TYPE_REPEAT) && (num == (EGIS0570_REPEAT_TOTAL - 1)));
|
|
||||||
|
|
||||||
return r;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Returns a bit for each frame on whether or not a finger has been detected.
|
|
||||||
* e.g. 00110 means that there is a finger in frame two and three.
|
|
||||||
*/
|
|
||||||
static char
|
|
||||||
postprocess_frames (FpDeviceEgis0570 *self, guint8 * img)
|
|
||||||
{
|
|
||||||
size_t mean[EGIS0570_IMGCOUNT] = {0, 0, 0, 0, 0};
|
|
||||||
|
|
||||||
if (!self->background)
|
|
||||||
{
|
|
||||||
self->background = g_malloc (EGIS0570_IMGWIDTH * EGIS0570_RFMGHEIGHT);
|
|
||||||
memset (self->background, 255, EGIS0570_IMGWIDTH * EGIS0570_RFMGHEIGHT);
|
|
||||||
|
|
||||||
for (size_t k = 0; k < EGIS0570_IMGCOUNT; k += 1)
|
|
||||||
{
|
|
||||||
guint8 * frame = &img[(k * EGIS0570_IMGSIZE) + EGIS0570_RFMDIS * EGIS0570_IMGWIDTH];
|
|
||||||
|
|
||||||
for (size_t i = 0; i < EGIS0570_IMGWIDTH * EGIS0570_RFMGHEIGHT; i += 1)
|
|
||||||
self->background[i] = MIN (self->background[i], frame[i]);
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (size_t k = 0; k < EGIS0570_IMGCOUNT; k += 1)
|
|
||||||
{
|
|
||||||
guint8 * frame = &img[(k * EGIS0570_IMGSIZE) + EGIS0570_RFMDIS * EGIS0570_IMGWIDTH];
|
|
||||||
|
|
||||||
for (size_t i = 0; i < EGIS0570_IMGWIDTH * EGIS0570_RFMGHEIGHT; i += 1)
|
|
||||||
{
|
|
||||||
if (frame[i] - EGIS0570_MARGIN > self->background[i])
|
|
||||||
frame[i] -= self->background[i];
|
|
||||||
else
|
|
||||||
frame[i] = 0;
|
|
||||||
|
|
||||||
mean[k] += frame[i];
|
|
||||||
}
|
|
||||||
|
|
||||||
mean[k] /= EGIS0570_IMGWIDTH * EGIS0570_RFMGHEIGHT;
|
|
||||||
}
|
|
||||||
|
|
||||||
char result = 0;
|
|
||||||
|
|
||||||
for (size_t k = 0; k < EGIS0570_IMGCOUNT; k += 1)
|
|
||||||
{
|
|
||||||
fp_dbg ("Finger status (picture number, mean) : %ld , %ld", k, mean[k]);
|
|
||||||
if (mean[k] > EGIS0570_MIN_MEAN)
|
|
||||||
result |= 1 << k;
|
|
||||||
}
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Device communication
|
|
||||||
*/
|
|
||||||
|
|
||||||
static void
|
|
||||||
data_resp_cb (FpiUsbTransfer *transfer, FpDevice *dev, gpointer user_data, GError *error)
|
|
||||||
{
|
|
||||||
unsigned char *stripdata;
|
|
||||||
gboolean end = FALSE;
|
|
||||||
FpImageDevice *img_self = FP_IMAGE_DEVICE (dev);
|
|
||||||
FpDeviceEgis0570 *self = FPI_DEVICE_EGIS0570 (dev);
|
|
||||||
|
|
||||||
if (error)
|
|
||||||
{
|
|
||||||
fpi_ssm_mark_failed (transfer->ssm, error);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
int where_finger_is = postprocess_frames (self, transfer->buffer);
|
|
||||||
|
|
||||||
if (where_finger_is > 0)
|
|
||||||
{
|
|
||||||
FpiImageDeviceState state;
|
|
||||||
|
|
||||||
fpi_image_device_report_finger_status (img_self, TRUE);
|
|
||||||
|
|
||||||
g_object_get (dev, "fpi-image-device-state", &state, NULL);
|
|
||||||
if (state == FPI_IMAGE_DEVICE_STATE_CAPTURE)
|
|
||||||
{
|
|
||||||
for (size_t k = 0; k < EGIS0570_IMGCOUNT; k += 1)
|
|
||||||
{
|
|
||||||
if (where_finger_is & (1 << k))
|
|
||||||
{
|
|
||||||
struct fpi_frame *stripe = g_malloc (EGIS0570_IMGWIDTH * EGIS0570_RFMGHEIGHT + sizeof (struct fpi_frame));
|
|
||||||
stripe->delta_x = 0;
|
|
||||||
stripe->delta_y = 0;
|
|
||||||
stripdata = stripe->data;
|
|
||||||
memcpy (stripdata, (transfer->buffer) + (((k) * EGIS0570_IMGSIZE) + EGIS0570_IMGWIDTH * EGIS0570_RFMDIS), EGIS0570_IMGWIDTH * EGIS0570_RFMGHEIGHT);
|
|
||||||
self->strips = g_slist_prepend (self->strips, stripe);
|
|
||||||
self->strips_len += 1;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
end = TRUE;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
end = TRUE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (end)
|
|
||||||
{
|
|
||||||
if (!self->stop && (self->strips_len > 0))
|
|
||||||
{
|
|
||||||
FpImage *img;
|
|
||||||
self->strips = g_slist_reverse (self->strips);
|
|
||||||
fpi_do_movement_estimation (&assembling_ctx, self->strips);
|
|
||||||
img = fpi_assemble_frames (&assembling_ctx, self->strips);
|
|
||||||
img->flags |= (FPI_IMAGE_COLORS_INVERTED | FPI_IMAGE_PARTIAL);
|
|
||||||
g_slist_free_full (self->strips, g_free);
|
|
||||||
self->strips = NULL;
|
|
||||||
self->strips_len = 0;
|
|
||||||
FpImage *resizeImage = fpi_image_resize (img, EGIS0570_RESIZE, EGIS0570_RESIZE);
|
|
||||||
fpi_image_device_image_captured (img_self, resizeImage);
|
|
||||||
}
|
|
||||||
|
|
||||||
fpi_image_device_report_finger_status (img_self, FALSE);
|
|
||||||
}
|
|
||||||
|
|
||||||
fpi_ssm_next_state (transfer->ssm);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
recv_data_resp (FpiSsm *ssm, FpDevice *dev)
|
|
||||||
{
|
|
||||||
FpiUsbTransfer *transfer = fpi_usb_transfer_new (dev);
|
|
||||||
|
|
||||||
fpi_usb_transfer_fill_bulk (transfer, EGIS0570_EPIN, EGIS0570_INPSIZE);
|
|
||||||
|
|
||||||
transfer->ssm = ssm;
|
|
||||||
transfer->short_is_error = TRUE;
|
|
||||||
|
|
||||||
fpi_usb_transfer_submit (transfer, EGIS0570_TIMEOUT, NULL, data_resp_cb, NULL);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
cmd_resp_cb (FpiUsbTransfer *transfer, FpDevice *dev, gpointer user_data, GError *error)
|
|
||||||
{
|
|
||||||
if (error)
|
|
||||||
fpi_ssm_mark_failed (transfer->ssm, error);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
recv_cmd_resp (FpiSsm *ssm, FpDevice *dev)
|
|
||||||
{
|
|
||||||
FpiUsbTransfer *transfer = fpi_usb_transfer_new (dev);
|
|
||||||
|
|
||||||
fpi_usb_transfer_fill_bulk (transfer, EGIS0570_EPIN, EGIS0570_PKTSIZE);
|
|
||||||
|
|
||||||
transfer->ssm = ssm;
|
|
||||||
|
|
||||||
fpi_usb_transfer_submit (transfer, EGIS0570_TIMEOUT, NULL, cmd_resp_cb, NULL);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
send_cmd_req (FpiSsm *ssm, FpDevice *dev, unsigned char *pkt)
|
|
||||||
{
|
|
||||||
FpiUsbTransfer *transfer = fpi_usb_transfer_new (dev);
|
|
||||||
|
|
||||||
fpi_usb_transfer_fill_bulk_full (transfer, EGIS0570_EPOUT, pkt, EGIS0570_PKTSIZE, NULL);
|
|
||||||
|
|
||||||
transfer->ssm = ssm;
|
|
||||||
transfer->short_is_error = TRUE;
|
|
||||||
|
|
||||||
fpi_usb_transfer_submit (transfer, EGIS0570_TIMEOUT, NULL, fpi_ssm_usb_transfer_cb, NULL);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* SSM States
|
|
||||||
*/
|
|
||||||
|
|
||||||
enum sm_states {
|
|
||||||
SM_INIT,
|
|
||||||
SM_START,
|
|
||||||
SM_REQ,
|
|
||||||
SM_RESP,
|
|
||||||
SM_REC_DATA,
|
|
||||||
SM_DONE,
|
|
||||||
SM_STATES_NUM
|
|
||||||
};
|
|
||||||
|
|
||||||
static void
|
|
||||||
ssm_run_state (FpiSsm *ssm, FpDevice *dev)
|
|
||||||
{
|
|
||||||
FpDeviceEgis0570 *self = FPI_DEVICE_EGIS0570 (dev);
|
|
||||||
FpImageDevice *img_dev = FP_IMAGE_DEVICE (dev);
|
|
||||||
|
|
||||||
switch (fpi_ssm_get_cur_state (ssm))
|
|
||||||
{
|
|
||||||
case SM_INIT:
|
|
||||||
self->pkt_type = PKT_TYPE_INIT;
|
|
||||||
fpi_ssm_next_state (ssm);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case SM_START:
|
|
||||||
if (self->stop)
|
|
||||||
{
|
|
||||||
fp_dbg ("deactivating, marking completed");
|
|
||||||
fpi_ssm_mark_completed (ssm);
|
|
||||||
fpi_image_device_deactivate_complete (img_dev, NULL);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
self->pkt_num = 0;
|
|
||||||
fpi_ssm_next_state (ssm);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case SM_REQ:
|
|
||||||
if (self->pkt_type == PKT_TYPE_INIT)
|
|
||||||
send_cmd_req (ssm, dev, init_pkts[self->pkt_num]);
|
|
||||||
else
|
|
||||||
send_cmd_req (ssm, dev, repeat_pkts[self->pkt_num]);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case SM_RESP:
|
|
||||||
if (is_last_pkt (dev) == FALSE)
|
|
||||||
{
|
|
||||||
recv_cmd_resp (ssm, dev);
|
|
||||||
self->pkt_num += 1;
|
|
||||||
fpi_ssm_jump_to_state (ssm, SM_REQ);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (self->pkt_type == PKT_TYPE_INIT)
|
|
||||||
self->pkt_type = PKT_TYPE_REPEAT;
|
|
||||||
|
|
||||||
fpi_ssm_next_state (ssm);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case SM_REC_DATA:
|
|
||||||
recv_data_resp (ssm, dev);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case SM_DONE:
|
|
||||||
fpi_ssm_jump_to_state (ssm, SM_START);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
g_assert_not_reached ();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Activation
|
|
||||||
*/
|
|
||||||
|
|
||||||
static void
|
|
||||||
loop_complete (FpiSsm *ssm, FpDevice *dev, GError *error)
|
|
||||||
{
|
|
||||||
FpImageDevice *img_dev = FP_IMAGE_DEVICE (dev);
|
|
||||||
FpDeviceEgis0570 *self = FPI_DEVICE_EGIS0570 (dev);
|
|
||||||
|
|
||||||
self->running = FALSE;
|
|
||||||
g_clear_pointer (&self->background, g_free);
|
|
||||||
|
|
||||||
if (error)
|
|
||||||
fpi_image_device_session_error (img_dev, error);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
dev_activate (FpImageDevice *dev)
|
|
||||||
{
|
|
||||||
FpDeviceEgis0570 *self = FPI_DEVICE_EGIS0570 (dev);
|
|
||||||
FpiSsm *ssm = fpi_ssm_new (FP_DEVICE (dev), ssm_run_state, SM_STATES_NUM);
|
|
||||||
|
|
||||||
self->stop = FALSE;
|
|
||||||
|
|
||||||
fpi_ssm_start (ssm, loop_complete);
|
|
||||||
|
|
||||||
self->running = TRUE;
|
|
||||||
|
|
||||||
fpi_image_device_activate_complete (dev, NULL);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Opening
|
|
||||||
*/
|
|
||||||
|
|
||||||
static void
|
|
||||||
dev_init (FpImageDevice *dev)
|
|
||||||
{
|
|
||||||
GError *error = NULL;
|
|
||||||
|
|
||||||
g_usb_device_claim_interface (fpi_device_get_usb_device (FP_DEVICE (dev)), 0, 0, &error);
|
|
||||||
|
|
||||||
fpi_image_device_open_complete (dev, error);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Closing
|
|
||||||
*/
|
|
||||||
|
|
||||||
static void
|
|
||||||
dev_deinit (FpImageDevice *dev)
|
|
||||||
{
|
|
||||||
GError *error = NULL;
|
|
||||||
|
|
||||||
g_usb_device_release_interface (fpi_device_get_usb_device (FP_DEVICE (dev)), 0, 0, &error);
|
|
||||||
|
|
||||||
fpi_image_device_close_complete (dev, error);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Deactivation
|
|
||||||
*/
|
|
||||||
|
|
||||||
static void
|
|
||||||
dev_deactivate (FpImageDevice *dev)
|
|
||||||
{
|
|
||||||
FpDeviceEgis0570 *self = FPI_DEVICE_EGIS0570 (dev);
|
|
||||||
|
|
||||||
if (self->running)
|
|
||||||
self->stop = TRUE;
|
|
||||||
else
|
|
||||||
fpi_image_device_deactivate_complete (dev, NULL);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Driver data
|
|
||||||
*/
|
|
||||||
|
|
||||||
static const FpIdEntry id_table[] = {
|
|
||||||
{ .vid = 0x1c7a, .pid = 0x0570, },
|
|
||||||
{ .vid = 0x1c7a, .pid = 0x0571, },
|
|
||||||
{ .vid = 0, .pid = 0, },
|
|
||||||
};
|
|
||||||
|
|
||||||
static void
|
|
||||||
fpi_device_egis0570_init (FpDeviceEgis0570 *self)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
fpi_device_egis0570_class_init (FpDeviceEgis0570Class *klass)
|
|
||||||
{
|
|
||||||
FpDeviceClass *dev_class = FP_DEVICE_CLASS (klass);
|
|
||||||
FpImageDeviceClass *img_class = FP_IMAGE_DEVICE_CLASS (klass);
|
|
||||||
|
|
||||||
dev_class->id = "egis0570";
|
|
||||||
dev_class->full_name = "Egis Technology Inc. (aka. LighTuning) 0570";
|
|
||||||
dev_class->type = FP_DEVICE_TYPE_USB;
|
|
||||||
dev_class->id_table = id_table;
|
|
||||||
dev_class->scan_type = FP_SCAN_TYPE_SWIPE;
|
|
||||||
|
|
||||||
img_class->img_open = dev_init;
|
|
||||||
img_class->img_close = dev_deinit;
|
|
||||||
img_class->activate = dev_activate;
|
|
||||||
img_class->deactivate = dev_deactivate;
|
|
||||||
|
|
||||||
img_class->img_width = EGIS0570_IMGWIDTH;
|
|
||||||
img_class->img_height = -1;
|
|
||||||
|
|
||||||
img_class->bz3_threshold = EGIS0570_BZ3_THRESHOLD; /* security issue */
|
|
||||||
}
|
|
|
@ -1,177 +0,0 @@
|
||||||
/*
|
|
||||||
* Egis Technology Inc. (aka. LighTuning) 0570 driver for libfprint
|
|
||||||
* Copyright (C) 2021 Maxim Kolesnikov <kolesnikov@svyazcom.ru>
|
|
||||||
* Copyright (C) 2021 Saeed/Ali Rk <saeed.ali.rahimi@gmail.com>
|
|
||||||
*
|
|
||||||
* This library is free software; you can redistribute it and/or
|
|
||||||
* modify it under the terms of the GNU Lesser General Public
|
|
||||||
* License as published by the Free Software Foundation; either
|
|
||||||
* version 2.1 of the License, or (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This library is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
||||||
* Lesser General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Lesser General Public
|
|
||||||
* License along with this library; if not, write to the Free Software
|
|
||||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef __EGIS0570_H
|
|
||||||
|
|
||||||
#define __EGIS0570_H 1
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Device data
|
|
||||||
*/
|
|
||||||
|
|
||||||
#define EGIS0570_CONF 1
|
|
||||||
#define EGIS0570_INTF 0
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Device endpoints
|
|
||||||
*/
|
|
||||||
|
|
||||||
#define EGIS0570_EPOUT 0x04 /* ( 4 | FPI_USB_ENDPOINT_OUT ) */
|
|
||||||
#define EGIS0570_EPIN 0x83 /* ( 3 | FPI_USB_ENDPOINT_IN ) */
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Initialization packets (7 bytes each)
|
|
||||||
*
|
|
||||||
* First 4 bytes are equivalent to string "EGIS", which must be just a company identificator
|
|
||||||
* Other 3 bytes are not recognized yet and may be not important, as they are always the same
|
|
||||||
|
|
||||||
* Answers for each packet contain 7 bytes again
|
|
||||||
* First 4 bytes are reversed "EGIS", which is "SIGE", which is company ID again
|
|
||||||
* Other 3 bytes are not recognized yet
|
|
||||||
* But there is a pattern.
|
|
||||||
* Sending last packet makes sensor return image
|
|
||||||
*/
|
|
||||||
|
|
||||||
#define EGIS0570_TIMEOUT 10000
|
|
||||||
#define EGIS0570_PKTSIZE 7
|
|
||||||
|
|
||||||
#define EGIS0570_INIT_TOTAL (sizeof ((init_pkts)) / sizeof ((init_pkts[0])))
|
|
||||||
|
|
||||||
static unsigned char init_pkts[][EGIS0570_PKTSIZE] =
|
|
||||||
{
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x20, 0x3f },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x58, 0x3f },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x21, 0x09 },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x57, 0x09 },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x22, 0x03 },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x56, 0x03 },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x23, 0x01 },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x55, 0x01 },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x24, 0x01 },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x54, 0x01 },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x16, 0x3e },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x09, 0x0b },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x14, 0x03 },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x15, 0x00 },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x02, 0x0f },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x10, 0x00 },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x11, 0x38 },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x12, 0x00 },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x13, 0x71 },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x03, 0x80 },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x00, 0x02, 0x80 },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x02, 0x2f },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x06, 0x00, 0xfe } /* image returned after this packet */
|
|
||||||
};
|
|
||||||
|
|
||||||
/* There is another Packet !
|
|
||||||
* That just Work the same !!
|
|
||||||
* And the Size is different !!!
|
|
||||||
*/
|
|
||||||
|
|
||||||
/*
|
|
||||||
|
|
||||||
#define EGIS0570_INIT_TOTAL2 (sizeof((init_pkts2)) / sizeof((init_pkts2[0])))
|
|
||||||
|
|
||||||
static unsigned char init_pkts2[][EGIS0570_PKTSIZE] =
|
|
||||||
{
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x10, 0x00},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x11, 0x38},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x12, 0x00},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x13, 0x71},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x20, 0x3f},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x58, 0x3f},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x21, 0x07},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x57, 0x07},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x22, 0x02},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x56, 0x02},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x23, 0x00},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x55, 0x00},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x24, 0x00},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x54, 0x00},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x25, 0x00},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x53, 0x00},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x15, 0x00},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x16, 0x3b},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x09, 0x0a},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x14, 0x00},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x02, 0x0f},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x03, 0x80},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x00, 0x02, 0x80},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x01, 0x02, 0x2f},
|
|
||||||
{0x45, 0x47, 0x49, 0x53, 0x06, 0x00, 0xfe}
|
|
||||||
};
|
|
||||||
|
|
||||||
*/
|
|
||||||
|
|
||||||
/*
|
|
||||||
* After sending initial packets device returns image data (32512 bytes)
|
|
||||||
* To ask device to send image data again, host needs to send four additional packets
|
|
||||||
* Further work is to repeatedly send four repeat packets and read image data
|
|
||||||
*/
|
|
||||||
|
|
||||||
#define EGIS0570_INPSIZE 32512
|
|
||||||
|
|
||||||
/* 5 image with captured in different time of size 114 * 57 = 6498
|
|
||||||
* 5 * 6498 = 32490 plus 22 extra unrecognized char size data
|
|
||||||
* Two continuous image in this 5 images may have time delay of less than 20ms
|
|
||||||
*/
|
|
||||||
|
|
||||||
#define EGIS0570_IMGSIZE 6498
|
|
||||||
#define EGIS0570_IMGWIDTH 114
|
|
||||||
#define EGIS0570_IMGHEIGHT 57
|
|
||||||
|
|
||||||
/* size of middle area that is used from each frame */
|
|
||||||
#define EGIS0570_RFMGHEIGHT 17
|
|
||||||
/* rows to ignore from top and bottom of the image*/
|
|
||||||
#define EGIS0570_RFMDIS (EGIS0570_IMGHEIGHT - EGIS0570_RFMGHEIGHT) / 2
|
|
||||||
#define EGIS0570_IMGCOUNT 5
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Image repeat request
|
|
||||||
* First 4 bytes are the same as in initialization packets
|
|
||||||
* Have no idea what the other 3 bytes mean
|
|
||||||
*/
|
|
||||||
|
|
||||||
#define EGIS0570_REPEAT_TOTAL (sizeof ((repeat_pkts)) / sizeof ((repeat_pkts[0])))
|
|
||||||
|
|
||||||
static unsigned char repeat_pkts[][EGIS0570_PKTSIZE] =
|
|
||||||
{
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x02, 0x0f },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x00, 0x02, 0x0f },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x01, 0x02, 0x2f },
|
|
||||||
{ 0x45, 0x47, 0x49, 0x53, 0x06, 0x00, 0xfe } /* image returned after this packet */
|
|
||||||
};
|
|
||||||
|
|
||||||
/*
|
|
||||||
* This sensor is small so I decided to reduce bz3_threshold from
|
|
||||||
* 40 to 10 to have more success to fail ratio
|
|
||||||
* Bozorth3 Algorithm seems not fine at the end
|
|
||||||
* foreget about security :))
|
|
||||||
*/
|
|
||||||
|
|
||||||
#define EGIS0570_BZ3_THRESHOLD 25 /* and even less What a joke */
|
|
||||||
|
|
||||||
#define EGIS0570_MIN_MEAN 20
|
|
||||||
#define EGIS0570_MARGIN 3
|
|
||||||
|
|
||||||
#define EGIS0570_RESIZE 2
|
|
||||||
|
|
||||||
#endif
|
|
|
@ -218,8 +218,6 @@ static const FpIdEntry elan_id_table[] = {
|
||||||
{.vid = ELAN_VEND_ID, .pid = 0x0c4d, .driver_data = ELAN_ALL_DEV},
|
{.vid = ELAN_VEND_ID, .pid = 0x0c4d, .driver_data = ELAN_ALL_DEV},
|
||||||
{.vid = ELAN_VEND_ID, .pid = 0x0c4f, .driver_data = ELAN_ALL_DEV},
|
{.vid = ELAN_VEND_ID, .pid = 0x0c4f, .driver_data = ELAN_ALL_DEV},
|
||||||
{.vid = ELAN_VEND_ID, .pid = 0x0c63, .driver_data = ELAN_ALL_DEV},
|
{.vid = ELAN_VEND_ID, .pid = 0x0c63, .driver_data = ELAN_ALL_DEV},
|
||||||
{.vid = ELAN_VEND_ID, .pid = 0x0c6e, .driver_data = ELAN_ALL_DEV},
|
|
||||||
{.vid = ELAN_VEND_ID, .pid = 0x0c58, .driver_data = ELAN_ALL_DEV},
|
|
||||||
{.vid = 0, .pid = 0, .driver_data = 0},
|
{.vid = 0, .pid = 0, .driver_data = 0},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -1,195 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright (C) 2021 Elan Microelectronics
|
|
||||||
*
|
|
||||||
* This library is free software; you can redistribute it and/or
|
|
||||||
* modify it under the terms of the GNU Lesser General Public
|
|
||||||
* License as published by the Free Software Foundation; either
|
|
||||||
* version 2.1 of the License, or (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This library is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
||||||
* Lesser General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Lesser General Public
|
|
||||||
* License along with this library; if not, write to the Free Software
|
|
||||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "fpi-device.h"
|
|
||||||
#include "fpi-ssm.h"
|
|
||||||
#include <libusb.h>
|
|
||||||
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
|
|
||||||
G_DECLARE_FINAL_TYPE (FpiDeviceElanmoc, fpi_device_elanmoc, FPI, DEVICE_ELANMOC, FpDevice)
|
|
||||||
|
|
||||||
#define ELAN_MOC_DRIVER_FULLNAME "Elan MOC Sensors"
|
|
||||||
#define ELAN_M0C_CMD_LEN 0x3
|
|
||||||
#define ELAN_EP_CMD_OUT (0x1 | LIBUSB_ENDPOINT_OUT)
|
|
||||||
#define ELAN_EP_CMD_IN (0x3 | LIBUSB_ENDPOINT_IN)
|
|
||||||
#define ELAN_EP_MOC_CMD_IN (0x4 | LIBUSB_ENDPOINT_IN)
|
|
||||||
#define ELAN_EP_IMG_IN (0x2 | LIBUSB_ENDPOINT_IN)
|
|
||||||
|
|
||||||
#define ELAN_MOC_CMD_TIMEOUT 5000
|
|
||||||
#define ELAN_MOC_CAL_RETRY 500
|
|
||||||
#define ELAN_MOC_ENROLL_TIMES 9
|
|
||||||
#define ELAN_MAX_USER_ID_LEN 92
|
|
||||||
#define ELAN_MAX_ENROLL_NUM 9
|
|
||||||
|
|
||||||
#define ELAN_MSG_VERIFY_ERR 0xfd
|
|
||||||
#define ELAN_MSG_DIRTY 0xfb
|
|
||||||
#define ELAN_MSG_AREA_NOT_ENOUGH 0xfe
|
|
||||||
#define ELAN_MSG_TOO_HIGH 0x41
|
|
||||||
#define ELAN_MSG_TOO_LEFT 0x42
|
|
||||||
#define ELAN_MSG_TOO_LOW 0x43
|
|
||||||
#define ELAN_MSG_TOO_RIGHT 0x44
|
|
||||||
#define ELAN_MSG_OK 0x00
|
|
||||||
|
|
||||||
#define ELAN_MAX_HDR_LEN 3
|
|
||||||
#define ELAN_USERDATE_SIZE (ELAN_MAX_USER_ID_LEN + 3)
|
|
||||||
|
|
||||||
#define ELAN_MSG_DRIVER_VERSION "1004"
|
|
||||||
|
|
||||||
struct elanmoc_cmd
|
|
||||||
{
|
|
||||||
unsigned char cmd_header[ELAN_MAX_HDR_LEN];
|
|
||||||
int cmd_len;
|
|
||||||
int resp_len;
|
|
||||||
};
|
|
||||||
|
|
||||||
static const struct elanmoc_cmd fw_ver_cmd = {
|
|
||||||
.cmd_header = {0x40, 0x19},
|
|
||||||
.cmd_len = 2,
|
|
||||||
.resp_len = 2,
|
|
||||||
};
|
|
||||||
|
|
||||||
static const struct elanmoc_cmd sensor_dim_cmd = {
|
|
||||||
.cmd_header = {0x00, 0x0c},
|
|
||||||
.cmd_len = 2,
|
|
||||||
.resp_len = 4,
|
|
||||||
};
|
|
||||||
|
|
||||||
static const struct elanmoc_cmd cal_status_cmd = {
|
|
||||||
.cmd_header = {0x40, 0xff, 0x00},
|
|
||||||
.cmd_len = 3,
|
|
||||||
.resp_len = 2,
|
|
||||||
};
|
|
||||||
|
|
||||||
static const struct elanmoc_cmd enrolled_number_cmd = {
|
|
||||||
.cmd_header = {0x40, 0xff, 0x04},
|
|
||||||
.cmd_len = 3,
|
|
||||||
.resp_len = 2,
|
|
||||||
};
|
|
||||||
|
|
||||||
static const struct elanmoc_cmd elanmoc_verify_cmd = {
|
|
||||||
.cmd_header = {0x40, 0xff, 0x73},
|
|
||||||
.cmd_len = 5,
|
|
||||||
.resp_len = 2,
|
|
||||||
};
|
|
||||||
|
|
||||||
static const struct elanmoc_cmd elanmoc_above_cmd = {
|
|
||||||
.cmd_header = {0x40, 0xff, 0x02},
|
|
||||||
.cmd_len = 3,
|
|
||||||
.resp_len = 0,
|
|
||||||
};
|
|
||||||
|
|
||||||
static const struct elanmoc_cmd elanmoc_enroll_cmd = {
|
|
||||||
.cmd_header = {0x40, 0xff, 0x01},
|
|
||||||
.cmd_len = 7,
|
|
||||||
.resp_len = 2,
|
|
||||||
};
|
|
||||||
|
|
||||||
static const struct elanmoc_cmd elanmoc_delete_cmd = {
|
|
||||||
.cmd_header = {0x40, 0xff, 0x13},
|
|
||||||
.cmd_len = 128,
|
|
||||||
.resp_len = 2,
|
|
||||||
};
|
|
||||||
|
|
||||||
static const struct elanmoc_cmd elanmoc_enroll_commit_cmd = {
|
|
||||||
.cmd_header = {0x40, 0xff, 0x11},
|
|
||||||
.cmd_len = 128,
|
|
||||||
.resp_len = 2,
|
|
||||||
};
|
|
||||||
|
|
||||||
static const struct elanmoc_cmd elanmoc_remove_all_cmd = {
|
|
||||||
.cmd_header = {0x40, 0xff, 0x98},
|
|
||||||
.cmd_len = 3,
|
|
||||||
.resp_len = 2,
|
|
||||||
};
|
|
||||||
|
|
||||||
static const struct elanmoc_cmd elanmoc_get_userid_cmd = {
|
|
||||||
.cmd_header = {0x43, 0x21, 0x00},
|
|
||||||
.cmd_len = 3,
|
|
||||||
.resp_len = 97,
|
|
||||||
};
|
|
||||||
|
|
||||||
static const struct elanmoc_cmd elanmoc_set_mod_cmd = {
|
|
||||||
.cmd_header = {0x40, 0xff, 0x14},
|
|
||||||
.cmd_len = 4,
|
|
||||||
.resp_len = 2,
|
|
||||||
};
|
|
||||||
|
|
||||||
static const struct elanmoc_cmd elanmoc_check_reenroll_cmd = {
|
|
||||||
.cmd_header = {0x40, 0xff, 0x22},
|
|
||||||
.cmd_len = 3 + ELAN_USERDATE_SIZE,
|
|
||||||
.resp_len = 2,
|
|
||||||
};
|
|
||||||
|
|
||||||
typedef void (*ElanCmdMsgCallback) (FpiDeviceElanmoc *self,
|
|
||||||
GError *error);
|
|
||||||
|
|
||||||
enum moc_enroll_states {
|
|
||||||
MOC_ENROLL_GET_ENROLLED_NUM,
|
|
||||||
MOC_ENROLL_REENROLL_CHECK,
|
|
||||||
MOC_ENROLL_WAIT_FINGER,
|
|
||||||
MOC_ENROLL_COMMIT_RESULT,
|
|
||||||
MOC_ENROLL_NUM_STATES,
|
|
||||||
};
|
|
||||||
|
|
||||||
enum moc_list_states {
|
|
||||||
MOC_LIST_GET_ENROLLED,
|
|
||||||
MOC_LIST_GET_FINGER,
|
|
||||||
MOC_LIST_NUM_STATES,
|
|
||||||
};
|
|
||||||
|
|
||||||
enum delete_states {
|
|
||||||
DELETE_SEND_CMD,
|
|
||||||
DELETE_NUM_STATES,
|
|
||||||
};
|
|
||||||
|
|
||||||
enum dev_init_states {
|
|
||||||
DEV_WAIT_READY,
|
|
||||||
DEV_SET_MODE,
|
|
||||||
DEV_GET_VER,
|
|
||||||
DEV_GET_DIM,
|
|
||||||
DEV_GET_ENROLLED,
|
|
||||||
DEV_INIT_STATES,
|
|
||||||
};
|
|
||||||
|
|
||||||
enum dev_exit_states {
|
|
||||||
DEV_EXIT_ABOVE,
|
|
||||||
DEV_EXIT_STATES,
|
|
||||||
};
|
|
||||||
|
|
||||||
struct _FpiDeviceElanmoc
|
|
||||||
{
|
|
||||||
FpDevice parent;
|
|
||||||
FpiSsm *task_ssm;
|
|
||||||
FpiSsm *cmd_ssm;
|
|
||||||
FpiUsbTransfer *cmd_transfer;
|
|
||||||
gboolean cmd_cancelable;
|
|
||||||
gsize cmd_len_in;
|
|
||||||
unsigned short fw_ver;
|
|
||||||
unsigned char x_trace;
|
|
||||||
unsigned char y_trace;
|
|
||||||
int num_frames;
|
|
||||||
int curr_enrolled;
|
|
||||||
int cancel_result;
|
|
||||||
int cmd_retry_cnt;
|
|
||||||
int list_index;
|
|
||||||
GPtrArray *list_result;
|
|
||||||
};
|
|
|
@ -1219,7 +1219,7 @@ elanspi_guess_image (FpiDeviceElanSpi *self, guint16 *raw_image)
|
||||||
|
|
||||||
sq_stddev /= (frame_width * frame_height);
|
sq_stddev /= (frame_width * frame_height);
|
||||||
|
|
||||||
fp_dbg ("<guess> stddev=%" G_GUINT64_FORMAT "d, ip=%d, is_fp=%d, is_empty=%d", sq_stddev, invalid_percent, is_fp, is_empty);
|
fp_dbg ("<guess> stddev=%ld, ip=%d, is_fp=%d, is_empty=%d", sq_stddev, invalid_percent, is_fp, is_empty);
|
||||||
|
|
||||||
if (invalid_percent < ELANSPI_MAX_REAL_INVALID_PERCENT)
|
if (invalid_percent < ELANSPI_MAX_REAL_INVALID_PERCENT)
|
||||||
is_fp += 1;
|
is_fp += 1;
|
||||||
|
|
|
@ -1320,7 +1320,6 @@ gx_fp_probe (FpDevice *device)
|
||||||
case 0x609C:
|
case 0x609C:
|
||||||
case 0x639C:
|
case 0x639C:
|
||||||
case 0x63AC:
|
case 0x63AC:
|
||||||
case 0x63BC:
|
|
||||||
case 0x6A94:
|
case 0x6A94:
|
||||||
self->max_enroll_stage = 12;
|
self->max_enroll_stage = 12;
|
||||||
break;
|
break;
|
||||||
|
@ -1544,7 +1543,6 @@ static const FpIdEntry id_table[] = {
|
||||||
{ .vid = 0x27c6, .pid = 0x60A2, },
|
{ .vid = 0x27c6, .pid = 0x60A2, },
|
||||||
{ .vid = 0x27c6, .pid = 0x639C, },
|
{ .vid = 0x27c6, .pid = 0x639C, },
|
||||||
{ .vid = 0x27c6, .pid = 0x63AC, },
|
{ .vid = 0x27c6, .pid = 0x63AC, },
|
||||||
{ .vid = 0x27c6, .pid = 0x63BC, },
|
|
||||||
{ .vid = 0x27c6, .pid = 0x6496, },
|
{ .vid = 0x27c6, .pid = 0x6496, },
|
||||||
{ .vid = 0x27c6, .pid = 0x6584, },
|
{ .vid = 0x27c6, .pid = 0x6584, },
|
||||||
{ .vid = 0x27c6, .pid = 0x658C, },
|
{ .vid = 0x27c6, .pid = 0x658C, },
|
||||||
|
@ -1567,7 +1565,6 @@ fpi_device_goodixmoc_class_init (FpiDeviceGoodixMocClass *klass)
|
||||||
dev_class->scan_type = FP_SCAN_TYPE_PRESS;
|
dev_class->scan_type = FP_SCAN_TYPE_PRESS;
|
||||||
dev_class->id_table = id_table;
|
dev_class->id_table = id_table;
|
||||||
dev_class->nr_enroll_stages = DEFAULT_ENROLL_SAMPLES;
|
dev_class->nr_enroll_stages = DEFAULT_ENROLL_SAMPLES;
|
||||||
dev_class->temp_hot_seconds = -1;
|
|
||||||
|
|
||||||
dev_class->open = gx_fp_init;
|
dev_class->open = gx_fp_init;
|
||||||
dev_class->close = gx_fp_exit;
|
dev_class->close = gx_fp_exit;
|
||||||
|
|
|
@ -40,9 +40,6 @@ static const FpIdEntry id_table[] = {
|
||||||
{ .vid = SYNAPTICS_VENDOR_ID, .pid = 0x0100, },
|
{ .vid = SYNAPTICS_VENDOR_ID, .pid = 0x0100, },
|
||||||
{ .vid = SYNAPTICS_VENDOR_ID, .pid = 0x00F0, },
|
{ .vid = SYNAPTICS_VENDOR_ID, .pid = 0x00F0, },
|
||||||
{ .vid = SYNAPTICS_VENDOR_ID, .pid = 0x0103, },
|
{ .vid = SYNAPTICS_VENDOR_ID, .pid = 0x0103, },
|
||||||
{ .vid = SYNAPTICS_VENDOR_ID, .pid = 0x0123, },
|
|
||||||
{ .vid = SYNAPTICS_VENDOR_ID, .pid = 0x0126, },
|
|
||||||
{ .vid = SYNAPTICS_VENDOR_ID, .pid = 0x0129, },
|
|
||||||
{ .vid = 0, .pid = 0, .driver_data = 0 }, /* terminating entry */
|
{ .vid = 0, .pid = 0, .driver_data = 0 }, /* terminating entry */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -201,16 +198,11 @@ cmd_interrupt_cb (FpiUsbTransfer *transfer,
|
||||||
GError *error)
|
GError *error)
|
||||||
{
|
{
|
||||||
g_debug ("interrupt transfer done");
|
g_debug ("interrupt transfer done");
|
||||||
fpi_device_critical_enter (device);
|
|
||||||
|
|
||||||
if (error)
|
if (error)
|
||||||
{
|
{
|
||||||
if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
|
if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
|
||||||
{
|
{
|
||||||
g_error_free (error);
|
g_error_free (error);
|
||||||
if (FPI_DEVICE_SYNAPTICS (device)->cmd_suspended)
|
|
||||||
fpi_ssm_jump_to_state (transfer->ssm, SYNAPTICS_CMD_SUSPENDED);
|
|
||||||
else
|
|
||||||
fpi_ssm_jump_to_state (transfer->ssm, SYNAPTICS_CMD_GET_RESP);
|
fpi_ssm_jump_to_state (transfer->ssm, SYNAPTICS_CMD_GET_RESP);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -272,9 +264,6 @@ synaptics_cmd_run_state (FpiSsm *ssm,
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case SYNAPTICS_CMD_WAIT_INTERRUPT:
|
case SYNAPTICS_CMD_WAIT_INTERRUPT:
|
||||||
/* Interruptions are permitted only during an interrupt transfer */
|
|
||||||
fpi_device_critical_leave (dev);
|
|
||||||
|
|
||||||
transfer = fpi_usb_transfer_new (dev);
|
transfer = fpi_usb_transfer_new (dev);
|
||||||
transfer->ssm = ssm;
|
transfer->ssm = ssm;
|
||||||
fpi_usb_transfer_fill_interrupt (transfer, USB_EP_INTERRUPT, USB_INTERRUPT_DATA_SIZE);
|
fpi_usb_transfer_fill_interrupt (transfer, USB_EP_INTERRUPT, USB_INTERRUPT_DATA_SIZE);
|
||||||
|
@ -302,17 +291,6 @@ synaptics_cmd_run_state (FpiSsm *ssm,
|
||||||
case SYNAPTICS_CMD_RESTART:
|
case SYNAPTICS_CMD_RESTART:
|
||||||
fpi_ssm_jump_to_state (ssm, SYNAPTICS_CMD_SEND_PENDING);
|
fpi_ssm_jump_to_state (ssm, SYNAPTICS_CMD_SEND_PENDING);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case SYNAPTICS_CMD_SUSPENDED:
|
|
||||||
/* The resume handler continues to the next state! */
|
|
||||||
fpi_device_critical_leave (dev);
|
|
||||||
fpi_device_suspend_complete (dev, NULL);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case SYNAPTICS_CMD_RESUME:
|
|
||||||
fpi_device_critical_enter (dev);
|
|
||||||
fpi_ssm_jump_to_state (ssm, SYNAPTICS_CMD_WAIT_INTERRUPT);
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -328,7 +306,6 @@ cmd_ssm_done (FpiSsm *ssm, FpDevice *dev, GError *error)
|
||||||
if (error || self->cmd_complete_on_removal)
|
if (error || self->cmd_complete_on_removal)
|
||||||
callback (self, NULL, error);
|
callback (self, NULL, error);
|
||||||
|
|
||||||
fpi_device_critical_leave (dev);
|
|
||||||
self->cmd_complete_on_removal = FALSE;
|
self->cmd_complete_on_removal = FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -438,7 +415,6 @@ synaptics_sensor_cmd (FpiDeviceSynaptics *self,
|
||||||
SYNAPTICS_CMD_NUM_STATES);
|
SYNAPTICS_CMD_NUM_STATES);
|
||||||
fpi_ssm_set_data (self->cmd_ssm, callback, NULL);
|
fpi_ssm_set_data (self->cmd_ssm, callback, NULL);
|
||||||
|
|
||||||
fpi_device_critical_enter (FP_DEVICE (self));
|
|
||||||
fpi_ssm_start (self->cmd_ssm, cmd_ssm_done);
|
fpi_ssm_start (self->cmd_ssm, cmd_ssm_done);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -534,12 +510,6 @@ verify_msg_cb (FpiDeviceSynaptics *self,
|
||||||
FpDevice *device = FP_DEVICE (self);
|
FpDevice *device = FP_DEVICE (self);
|
||||||
bmkt_verify_resp_t *verify_resp;
|
bmkt_verify_resp_t *verify_resp;
|
||||||
|
|
||||||
if (self->action_starting)
|
|
||||||
{
|
|
||||||
fpi_device_critical_leave (device);
|
|
||||||
self->action_starting = FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (error)
|
if (error)
|
||||||
{
|
{
|
||||||
fpi_device_verify_complete (device, error);
|
fpi_device_verify_complete (device, error);
|
||||||
|
@ -632,8 +602,6 @@ verify (FpDevice *device)
|
||||||
|
|
||||||
G_DEBUG_HERE ();
|
G_DEBUG_HERE ();
|
||||||
|
|
||||||
self->action_starting = TRUE;
|
|
||||||
fpi_device_critical_enter (device);
|
|
||||||
synaptics_sensor_cmd (self, 0, BMKT_CMD_VERIFY_USER, user_id, user_id_len, verify_msg_cb);
|
synaptics_sensor_cmd (self, 0, BMKT_CMD_VERIFY_USER, user_id, user_id_len, verify_msg_cb);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -661,12 +629,6 @@ identify_msg_cb (FpiDeviceSynaptics *self,
|
||||||
{
|
{
|
||||||
FpDevice *device = FP_DEVICE (self);
|
FpDevice *device = FP_DEVICE (self);
|
||||||
|
|
||||||
if (self->action_starting)
|
|
||||||
{
|
|
||||||
fpi_device_critical_leave (device);
|
|
||||||
self->action_starting = FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (error)
|
if (error)
|
||||||
{
|
{
|
||||||
fpi_device_identify_complete (device, error);
|
fpi_device_identify_complete (device, error);
|
||||||
|
@ -755,25 +717,6 @@ identify_msg_cb (FpiDeviceSynaptics *self,
|
||||||
static void
|
static void
|
||||||
identify (FpDevice *device)
|
identify (FpDevice *device)
|
||||||
{
|
{
|
||||||
GPtrArray *prints = NULL;
|
|
||||||
FpiDeviceSynaptics *self = FPI_DEVICE_SYNAPTICS (device);
|
|
||||||
|
|
||||||
fpi_device_get_identify_data (device, &prints);
|
|
||||||
|
|
||||||
/* Identify over no prints does not work for synaptics.
|
|
||||||
* This *may* make sense for other devices though, as identify may return
|
|
||||||
* a matched print even if it is not in the list of prints.
|
|
||||||
*/
|
|
||||||
if (prints->len == 0)
|
|
||||||
{
|
|
||||||
fpi_device_identify_report (device, NULL, NULL, NULL);
|
|
||||||
fpi_device_identify_complete (device, NULL);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
self->action_starting = TRUE;
|
|
||||||
fpi_device_critical_enter (device);
|
|
||||||
|
|
||||||
init_identify_msg (device);
|
init_identify_msg (device);
|
||||||
compose_and_send_identify_msg (device);
|
compose_and_send_identify_msg (device);
|
||||||
}
|
}
|
||||||
|
@ -876,12 +819,6 @@ enroll_msg_cb (FpiDeviceSynaptics *self,
|
||||||
FpDevice *device = FP_DEVICE (self);
|
FpDevice *device = FP_DEVICE (self);
|
||||||
bmkt_enroll_resp_t *enroll_resp;
|
bmkt_enroll_resp_t *enroll_resp;
|
||||||
|
|
||||||
if (self->action_starting)
|
|
||||||
{
|
|
||||||
fpi_device_critical_leave (device);
|
|
||||||
self->action_starting = FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (error)
|
if (error)
|
||||||
{
|
{
|
||||||
fpi_device_enroll_complete (device, NULL, error);
|
fpi_device_enroll_complete (device, NULL, error);
|
||||||
|
@ -1028,9 +965,6 @@ enroll (FpDevice *device)
|
||||||
payload[1] = finger;
|
payload[1] = finger;
|
||||||
memcpy (payload + 2, user_id, user_id_len);
|
memcpy (payload + 2, user_id, user_id_len);
|
||||||
|
|
||||||
self->action_starting = TRUE;
|
|
||||||
fpi_device_critical_enter (device);
|
|
||||||
|
|
||||||
synaptics_sensor_cmd (self, 0, BMKT_CMD_ENROLL_USER, payload, user_id_len + 2, enroll_msg_cb);
|
synaptics_sensor_cmd (self, 0, BMKT_CMD_ENROLL_USER, payload, user_id_len + 2, enroll_msg_cb);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1044,7 +978,6 @@ delete_msg_cb (FpiDeviceSynaptics *self,
|
||||||
|
|
||||||
if (error)
|
if (error)
|
||||||
{
|
{
|
||||||
fpi_device_critical_leave (device);
|
|
||||||
fpi_device_delete_complete (device, error);
|
fpi_device_delete_complete (device, error);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -1059,7 +992,6 @@ delete_msg_cb (FpiDeviceSynaptics *self,
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case BMKT_RSP_DEL_USER_FP_FAIL:
|
case BMKT_RSP_DEL_USER_FP_FAIL:
|
||||||
fpi_device_critical_leave (device);
|
|
||||||
if (resp->result == BMKT_FP_DATABASE_NO_RECORD_EXISTS ||
|
if (resp->result == BMKT_FP_DATABASE_NO_RECORD_EXISTS ||
|
||||||
resp->result == BMKT_FP_DATABASE_EMPTY)
|
resp->result == BMKT_FP_DATABASE_EMPTY)
|
||||||
{
|
{
|
||||||
|
@ -1076,7 +1008,6 @@ delete_msg_cb (FpiDeviceSynaptics *self,
|
||||||
|
|
||||||
case BMKT_RSP_DEL_USER_FP_OK:
|
case BMKT_RSP_DEL_USER_FP_OK:
|
||||||
fp_info ("Successfully deleted enrolled user");
|
fp_info ("Successfully deleted enrolled user");
|
||||||
fpi_device_critical_leave (device);
|
|
||||||
fpi_device_delete_complete (device, NULL);
|
fpi_device_delete_complete (device, NULL);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -1111,7 +1042,6 @@ delete_print (FpDevice *device)
|
||||||
payload[0] = finger;
|
payload[0] = finger;
|
||||||
memcpy (payload + 1, user_id, user_id_len);
|
memcpy (payload + 1, user_id, user_id_len);
|
||||||
|
|
||||||
fpi_device_critical_enter (device);
|
|
||||||
synaptics_sensor_cmd (self, 0, BMKT_CMD_DEL_USER_FP, payload, user_id_len + 1, delete_msg_cb);
|
synaptics_sensor_cmd (self, 0, BMKT_CMD_DEL_USER_FP, payload, user_id_len + 1, delete_msg_cb);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1351,12 +1281,8 @@ fps_deinit_cb (FpiDeviceSynaptics *self,
|
||||||
bmkt_response_t *resp,
|
bmkt_response_t *resp,
|
||||||
GError *error)
|
GError *error)
|
||||||
{
|
{
|
||||||
g_autoptr(GError) err = NULL;
|
|
||||||
|
|
||||||
/* Release usb interface */
|
/* Release usb interface */
|
||||||
g_usb_device_release_interface (fpi_device_get_usb_device (FP_DEVICE (self)), 0, 0, &err);
|
g_usb_device_release_interface (fpi_device_get_usb_device (FP_DEVICE (self)), 0, 0, &error);
|
||||||
if (!error)
|
|
||||||
error = g_steal_pointer (&err);
|
|
||||||
|
|
||||||
g_clear_object (&self->interrupt_cancellable);
|
g_clear_object (&self->interrupt_cancellable);
|
||||||
|
|
||||||
|
@ -1427,59 +1353,6 @@ cancel (FpDevice *dev)
|
||||||
self->interrupt_cancellable = g_cancellable_new ();
|
self->interrupt_cancellable = g_cancellable_new ();
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
suspend (FpDevice *dev)
|
|
||||||
{
|
|
||||||
FpiDeviceSynaptics *self = FPI_DEVICE_SYNAPTICS (dev);
|
|
||||||
FpiDeviceAction action = fpi_device_get_current_action (dev);
|
|
||||||
|
|
||||||
g_debug ("got suspend request");
|
|
||||||
|
|
||||||
if (action != FPI_DEVICE_ACTION_VERIFY && action != FPI_DEVICE_ACTION_IDENTIFY)
|
|
||||||
{
|
|
||||||
fpi_device_suspend_complete (dev, fpi_device_error_new (FP_DEVICE_ERROR_NOT_SUPPORTED));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* We are guaranteed to have a cmd_ssm running at this time. */
|
|
||||||
g_assert (self->cmd_ssm);
|
|
||||||
g_assert (fpi_ssm_get_cur_state (self->cmd_ssm) == SYNAPTICS_CMD_WAIT_INTERRUPT);
|
|
||||||
self->cmd_suspended = TRUE;
|
|
||||||
|
|
||||||
/* Cancel the current transfer.
|
|
||||||
* The CMD SSM will go into the suspend state and signal readyness. */
|
|
||||||
g_cancellable_cancel (self->interrupt_cancellable);
|
|
||||||
g_clear_object (&self->interrupt_cancellable);
|
|
||||||
self->interrupt_cancellable = g_cancellable_new ();
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
resume (FpDevice *dev)
|
|
||||||
{
|
|
||||||
FpiDeviceSynaptics *self = FPI_DEVICE_SYNAPTICS (dev);
|
|
||||||
FpiDeviceAction action = fpi_device_get_current_action (dev);
|
|
||||||
|
|
||||||
g_debug ("got resume request");
|
|
||||||
|
|
||||||
if (action != FPI_DEVICE_ACTION_VERIFY && action != FPI_DEVICE_ACTION_IDENTIFY)
|
|
||||||
{
|
|
||||||
g_assert_not_reached ();
|
|
||||||
fpi_device_resume_complete (dev, fpi_device_error_new (FP_DEVICE_ERROR_NOT_SUPPORTED));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* We must have a suspended cmd_ssm at this point */
|
|
||||||
g_assert (self->cmd_ssm);
|
|
||||||
g_assert (self->cmd_suspended);
|
|
||||||
g_assert (fpi_ssm_get_cur_state (self->cmd_ssm) == SYNAPTICS_CMD_SUSPENDED);
|
|
||||||
self->cmd_suspended = FALSE;
|
|
||||||
|
|
||||||
/* Restart interrupt transfer. */
|
|
||||||
fpi_ssm_jump_to_state (self->cmd_ssm, SYNAPTICS_CMD_RESUME);
|
|
||||||
|
|
||||||
fpi_device_resume_complete (dev, NULL);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
fpi_device_synaptics_init (FpiDeviceSynaptics *self)
|
fpi_device_synaptics_init (FpiDeviceSynaptics *self)
|
||||||
{
|
{
|
||||||
|
@ -1497,7 +1370,6 @@ fpi_device_synaptics_class_init (FpiDeviceSynapticsClass *klass)
|
||||||
dev_class->scan_type = FP_SCAN_TYPE_PRESS;
|
dev_class->scan_type = FP_SCAN_TYPE_PRESS;
|
||||||
dev_class->id_table = id_table;
|
dev_class->id_table = id_table;
|
||||||
dev_class->nr_enroll_stages = ENROLL_SAMPLES;
|
dev_class->nr_enroll_stages = ENROLL_SAMPLES;
|
||||||
dev_class->temp_hot_seconds = -1;
|
|
||||||
|
|
||||||
dev_class->open = dev_init;
|
dev_class->open = dev_init;
|
||||||
dev_class->close = dev_exit;
|
dev_class->close = dev_exit;
|
||||||
|
@ -1508,8 +1380,6 @@ fpi_device_synaptics_class_init (FpiDeviceSynapticsClass *klass)
|
||||||
dev_class->delete = delete_print;
|
dev_class->delete = delete_print;
|
||||||
dev_class->clear_storage = clear_storage;
|
dev_class->clear_storage = clear_storage;
|
||||||
dev_class->cancel = cancel;
|
dev_class->cancel = cancel;
|
||||||
dev_class->suspend = suspend;
|
|
||||||
dev_class->resume = resume;
|
|
||||||
|
|
||||||
fpi_device_class_auto_initialize_features (dev_class);
|
fpi_device_class_auto_initialize_features (dev_class);
|
||||||
}
|
}
|
||||||
|
|
|
@ -93,8 +93,6 @@ typedef enum {
|
||||||
SYNAPTICS_CMD_WAIT_INTERRUPT,
|
SYNAPTICS_CMD_WAIT_INTERRUPT,
|
||||||
SYNAPTICS_CMD_SEND_ASYNC,
|
SYNAPTICS_CMD_SEND_ASYNC,
|
||||||
SYNAPTICS_CMD_RESTART,
|
SYNAPTICS_CMD_RESTART,
|
||||||
SYNAPTICS_CMD_SUSPENDED,
|
|
||||||
SYNAPTICS_CMD_RESUME,
|
|
||||||
SYNAPTICS_CMD_NUM_STATES,
|
SYNAPTICS_CMD_NUM_STATES,
|
||||||
} SynapticsCmdState;
|
} SynapticsCmdState;
|
||||||
|
|
||||||
|
@ -112,12 +110,10 @@ struct _FpiDeviceSynaptics
|
||||||
FpiSsm *cmd_ssm;
|
FpiSsm *cmd_ssm;
|
||||||
FpiUsbTransfer *cmd_pending_transfer;
|
FpiUsbTransfer *cmd_pending_transfer;
|
||||||
gboolean cmd_complete_on_removal;
|
gboolean cmd_complete_on_removal;
|
||||||
gboolean cmd_suspended;
|
|
||||||
guint8 id_idx;
|
guint8 id_idx;
|
||||||
|
|
||||||
bmkt_sensor_version_t mis_version;
|
bmkt_sensor_version_t mis_version;
|
||||||
|
|
||||||
gboolean action_starting;
|
|
||||||
GCancellable *interrupt_cancellable;
|
GCancellable *interrupt_cancellable;
|
||||||
|
|
||||||
gint enroll_stage;
|
gint enroll_stage;
|
||||||
|
|
|
@ -79,7 +79,7 @@ struct _FpiDeviceUpeksonly
|
||||||
int num_flying;
|
int num_flying;
|
||||||
|
|
||||||
GSList *rows;
|
GSList *rows;
|
||||||
unsigned num_rows;
|
size_t num_rows;
|
||||||
unsigned char *rowbuf;
|
unsigned char *rowbuf;
|
||||||
int rowbuf_offset;
|
int rowbuf_offset;
|
||||||
|
|
||||||
|
@ -215,7 +215,7 @@ handoff_img (FpImageDevice *dev)
|
||||||
|
|
||||||
self->rows = g_slist_reverse (self->rows);
|
self->rows = g_slist_reverse (self->rows);
|
||||||
|
|
||||||
fp_dbg ("%u rows", self->num_rows);
|
fp_dbg ("%lu rows", self->num_rows);
|
||||||
img = fpi_assemble_lines (&self->assembling_ctx, self->rows, self->num_rows);
|
img = fpi_assemble_lines (&self->assembling_ctx, self->rows, self->num_rows);
|
||||||
|
|
||||||
g_slist_free_full (self->rows, g_free);
|
g_slist_free_full (self->rows, g_free);
|
||||||
|
@ -295,7 +295,7 @@ row_complete (FpImageDevice *dev)
|
||||||
if (self->num_blank > FINGER_REMOVED_THRESHOLD)
|
if (self->num_blank > FINGER_REMOVED_THRESHOLD)
|
||||||
{
|
{
|
||||||
self->finger_state = FINGER_REMOVED;
|
self->finger_state = FINGER_REMOVED;
|
||||||
fp_dbg ("detected finger removal. Blank rows: %d, Full rows: %u",
|
fp_dbg ("detected finger removal. Blank rows: %d, Full rows: %lu",
|
||||||
self->num_blank, self->num_rows);
|
self->num_blank, self->num_rows);
|
||||||
handoff_img (dev);
|
handoff_img (dev);
|
||||||
return;
|
return;
|
||||||
|
|
|
@ -411,7 +411,7 @@ dev_init (FpImageDevice *dev)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
fp_err ("Device variant %" G_GUINT64_FORMAT " is not known", driver_data);
|
fp_err ("Device variant %lu is not known", driver_data);
|
||||||
g_assert_not_reached ();
|
g_assert_not_reached ();
|
||||||
fpi_image_device_open_complete (dev, fpi_device_error_new (FP_DEVICE_ERROR_GENERAL));
|
fpi_image_device_open_complete (dev, fpi_device_error_new (FP_DEVICE_ERROR_GENERAL));
|
||||||
return;
|
return;
|
||||||
|
|
|
@ -221,7 +221,7 @@ capture_read_data_cb (FpiUsbTransfer *transfer, FpDevice *device,
|
||||||
if (response_size > transfer->actual_length)
|
if (response_size > transfer->actual_length)
|
||||||
{
|
{
|
||||||
fp_dbg ("response_size is %lu, actual_length is %d",
|
fp_dbg ("response_size is %lu, actual_length is %d",
|
||||||
(gulong) response_size, (gint) transfer->actual_length);
|
response_size, (gint) transfer->actual_length);
|
||||||
fp_dbg ("Waiting for rest of transfer");
|
fp_dbg ("Waiting for rest of transfer");
|
||||||
BUG_ON (self->response_rest);
|
BUG_ON (self->response_rest);
|
||||||
self->response_rest = response_size - transfer->actual_length;
|
self->response_rest = response_size - transfer->actual_length;
|
||||||
|
@ -309,7 +309,7 @@ capture_read_data_cb (FpiUsbTransfer *transfer, FpDevice *device,
|
||||||
data);
|
data);
|
||||||
BUG_ON (self->image_size != IMAGE_SIZE);
|
BUG_ON (self->image_size != IMAGE_SIZE);
|
||||||
fp_dbg ("Image size is %lu",
|
fp_dbg ("Image size is %lu",
|
||||||
(gulong) self->image_size);
|
self->image_size);
|
||||||
img = fp_image_new (IMAGE_WIDTH, IMAGE_HEIGHT);
|
img = fp_image_new (IMAGE_WIDTH, IMAGE_HEIGHT);
|
||||||
img->flags |= FPI_IMAGE_PARTIAL;
|
img->flags |= FPI_IMAGE_PARTIAL;
|
||||||
memcpy (img->data, self->image_bits,
|
memcpy (img->data, self->image_bits,
|
||||||
|
|
|
@ -366,7 +366,7 @@ read_msg_cb (FpiUsbTransfer *transfer, FpDevice *device,
|
||||||
fp_err ("async msg read too short (%d)",
|
fp_err ("async msg read too short (%d)",
|
||||||
(gint) transfer->actual_length);
|
(gint) transfer->actual_length);
|
||||||
error = fpi_device_error_new_msg (FP_DEVICE_ERROR_PROTO,
|
error = fpi_device_error_new_msg (FP_DEVICE_ERROR_PROTO,
|
||||||
"Packet from device was too short (%" G_GSSIZE_FORMAT ")",
|
"Packet from device was too short (%lu)",
|
||||||
transfer->actual_length);
|
transfer->actual_length);
|
||||||
goto err;
|
goto err;
|
||||||
}
|
}
|
||||||
|
@ -993,7 +993,7 @@ e_handle_resp00 (FpDevice *dev, unsigned char *data,
|
||||||
|
|
||||||
if (data_len != 14)
|
if (data_len != 14)
|
||||||
{
|
{
|
||||||
fp_err ("received 3001 poll response of %" G_GSIZE_FORMAT " bytes?", data_len);
|
fp_err ("received 3001 poll response of %lu bytes?", data_len);
|
||||||
do_enroll_stop (dev, NULL,
|
do_enroll_stop (dev, NULL,
|
||||||
fpi_device_error_new_msg (FP_DEVICE_ERROR_PROTO,
|
fpi_device_error_new_msg (FP_DEVICE_ERROR_PROTO,
|
||||||
"received 3001 response with wrong length"));
|
"received 3001 response with wrong length"));
|
||||||
|
@ -1090,7 +1090,7 @@ e_handle_resp02 (FpDevice *dev, unsigned char *data,
|
||||||
|
|
||||||
if (data_len < sizeof (scan_comp))
|
if (data_len < sizeof (scan_comp))
|
||||||
{
|
{
|
||||||
fp_err ("fingerprint data too short (%" G_GSIZE_FORMAT "u bytes)", data_len);
|
fp_err ("fingerprint data too short (%lu bytes)", data_len);
|
||||||
error = fpi_device_error_new_msg (FP_DEVICE_ERROR_PROTO, "fingerprint data too short");
|
error = fpi_device_error_new_msg (FP_DEVICE_ERROR_PROTO, "fingerprint data too short");
|
||||||
}
|
}
|
||||||
else if (memcmp (data, scan_comp, sizeof (scan_comp)) != 0)
|
else if (memcmp (data, scan_comp, sizeof (scan_comp)) != 0)
|
||||||
|
@ -1318,7 +1318,7 @@ v_handle_resp00 (FpDevice *dev, unsigned char *data,
|
||||||
|
|
||||||
if (data_len != 14)
|
if (data_len != 14)
|
||||||
{
|
{
|
||||||
fp_warn ("received 3001 poll response of %" G_GSIZE_FORMAT "u bytes?", data_len);
|
fp_warn ("received 3001 poll response of %lu bytes?", data_len);
|
||||||
error = fpi_device_error_new (FP_DEVICE_ERROR_PROTO);
|
error = fpi_device_error_new (FP_DEVICE_ERROR_PROTO);
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
|
@ -360,7 +360,7 @@ start_irq_handler (FpImageDevice *dev)
|
||||||
transfer = fpi_usb_transfer_new (FP_DEVICE (dev));
|
transfer = fpi_usb_transfer_new (FP_DEVICE (dev));
|
||||||
transfer->ssm = NULL;
|
transfer->ssm = NULL;
|
||||||
transfer->short_is_error = TRUE;
|
transfer->short_is_error = TRUE;
|
||||||
fpi_usb_transfer_fill_interrupt (transfer,
|
fpi_usb_transfer_fill_bulk (transfer,
|
||||||
EP_INTR,
|
EP_INTR,
|
||||||
IRQ_LENGTH);
|
IRQ_LENGTH);
|
||||||
fpi_usb_transfer_submit (transfer, 0, self->irq_cancellable, irq_handler, NULL);
|
fpi_usb_transfer_submit (transfer, 0, self->irq_cancellable, irq_handler, NULL);
|
||||||
|
|
|
@ -67,17 +67,12 @@ dev_identify (FpDevice *dev)
|
||||||
new_scan,
|
new_scan,
|
||||||
(GEqualFunc) fp_print_equal,
|
(GEqualFunc) fp_print_equal,
|
||||||
NULL))
|
NULL))
|
||||||
{
|
error = fpi_device_error_new (FP_DEVICE_ERROR_DATA_NOT_FOUND);
|
||||||
match = FALSE;
|
|
||||||
g_clear_object (&new_scan);
|
|
||||||
}
|
|
||||||
else if (g_ptr_array_find_with_equal_func (prints,
|
else if (g_ptr_array_find_with_equal_func (prints,
|
||||||
new_scan,
|
new_scan,
|
||||||
(GEqualFunc) fp_print_equal,
|
(GEqualFunc) fp_print_equal,
|
||||||
&idx))
|
&idx))
|
||||||
{
|
|
||||||
match = g_ptr_array_index (prints, idx);
|
match = g_ptr_array_index (prints, idx);
|
||||||
}
|
|
||||||
|
|
||||||
if (!self->match_reported)
|
if (!self->match_reported)
|
||||||
{
|
{
|
||||||
|
|
|
@ -150,8 +150,6 @@ process_cmds (FpDeviceVirtualDevice * self,
|
||||||
char **scan_id,
|
char **scan_id,
|
||||||
GError **error)
|
GError **error)
|
||||||
{
|
{
|
||||||
gboolean removed;
|
|
||||||
|
|
||||||
if (g_cancellable_is_cancelled (self->cancellable) ||
|
if (g_cancellable_is_cancelled (self->cancellable) ||
|
||||||
(fpi_device_get_current_action (FP_DEVICE (self)) != FPI_DEVICE_ACTION_NONE &&
|
(fpi_device_get_current_action (FP_DEVICE (self)) != FPI_DEVICE_ACTION_NONE &&
|
||||||
g_cancellable_is_cancelled (fpi_device_get_cancellable (FP_DEVICE (self)))))
|
g_cancellable_is_cancelled (fpi_device_get_cancellable (FP_DEVICE (self)))))
|
||||||
|
@ -194,7 +192,7 @@ process_cmds (FpDeviceVirtualDevice * self,
|
||||||
{
|
{
|
||||||
guint64 sleep_ms = g_ascii_strtoull (cmd + strlen (SLEEP_CMD_PREFIX), NULL, 10);
|
guint64 sleep_ms = g_ascii_strtoull (cmd + strlen (SLEEP_CMD_PREFIX), NULL, 10);
|
||||||
|
|
||||||
g_debug ("Sleeping %" G_GUINT64_FORMAT "ms", sleep_ms);
|
g_debug ("Sleeping %lums", sleep_ms);
|
||||||
self->sleep_timeout_id = g_timeout_add (sleep_ms, sleep_timeout_cb, self);
|
self->sleep_timeout_id = g_timeout_add (sleep_ms, sleep_timeout_cb, self);
|
||||||
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
@ -252,10 +250,7 @@ process_cmds (FpDeviceVirtualDevice * self,
|
||||||
if (self->ignore_wait)
|
if (self->ignore_wait)
|
||||||
return TRUE;
|
return TRUE;
|
||||||
|
|
||||||
g_object_get (self, "removed", &removed, NULL);
|
|
||||||
|
|
||||||
g_assert (self->wait_command_id == 0);
|
g_assert (self->wait_command_id == 0);
|
||||||
if (!scan || removed)
|
|
||||||
self->wait_command_id = g_timeout_add (500, wait_for_command_timeout, self);
|
self->wait_command_id = g_timeout_add (500, wait_for_command_timeout, self);
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
@ -280,7 +275,7 @@ recv_instruction_cb (GObject *source_object,
|
||||||
gsize bytes;
|
gsize bytes;
|
||||||
|
|
||||||
bytes = fpi_device_virtual_listener_read_finish (listener, res, &error);
|
bytes = fpi_device_virtual_listener_read_finish (listener, res, &error);
|
||||||
fp_dbg ("Got instructions of length %" G_GSIZE_FORMAT, bytes);
|
fp_dbg ("Got instructions of length %ld", bytes);
|
||||||
|
|
||||||
if (error)
|
if (error)
|
||||||
{
|
{
|
||||||
|
@ -309,7 +304,6 @@ recv_instruction_cb (GObject *source_object,
|
||||||
else if (g_str_has_prefix (cmd, UNPLUG_CMD))
|
else if (g_str_has_prefix (cmd, UNPLUG_CMD))
|
||||||
{
|
{
|
||||||
fpi_device_remove (FP_DEVICE (self));
|
fpi_device_remove (FP_DEVICE (self));
|
||||||
maybe_continue_current_action (self);
|
|
||||||
}
|
}
|
||||||
else if (g_str_has_prefix (cmd, SET_ENROLL_STAGES_PREFIX))
|
else if (g_str_has_prefix (cmd, SET_ENROLL_STAGES_PREFIX))
|
||||||
{
|
{
|
||||||
|
@ -543,7 +537,7 @@ dev_verify (FpDevice *dev)
|
||||||
|
|
||||||
if (self->prints_storage && !g_hash_table_contains (self->prints_storage, scan_id))
|
if (self->prints_storage && !g_hash_table_contains (self->prints_storage, scan_id))
|
||||||
{
|
{
|
||||||
g_clear_object (&new_scan);
|
error = fpi_device_error_new (FP_DEVICE_ERROR_DATA_NOT_FOUND);
|
||||||
success = FALSE;
|
success = FALSE;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
|
676
libfprint/drivers/virtual-sdcp.c
Normal file
676
libfprint/drivers/virtual-sdcp.c
Normal file
|
@ -0,0 +1,676 @@
|
||||||
|
/*
|
||||||
|
* Virtual driver for SDCP device debugging
|
||||||
|
*
|
||||||
|
* Copyright (C) 2020 Benjamin Berg <bberg@redhat.com>
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2.1 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This is a virtual test driver to test the basic SDCP functionality.
|
||||||
|
* It uses the test binaries from Microsoft, which were extended to allow
|
||||||
|
* a simple chat with the device.
|
||||||
|
* The environment variable contains the to be executed binary including
|
||||||
|
* arguments. This binary should be compiled from the code in
|
||||||
|
* https://github.com/Microsoft/SecureDeviceConnectionProtocol
|
||||||
|
* or, until it is merged upstream
|
||||||
|
* https://github.com/benzea/SecureDeviceConnectionProtocol
|
||||||
|
*
|
||||||
|
* Note that using this as an external executable has the advantage that we
|
||||||
|
* do not need to link against mbedtls or any other crypto library.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define FP_COMPONENT "virtual_sdcp"
|
||||||
|
|
||||||
|
#include "fpi-log.h"
|
||||||
|
#include "fpi-ssm.h"
|
||||||
|
|
||||||
|
#include "../fpi-sdcp-device.h"
|
||||||
|
|
||||||
|
#include <glib/gstdio.h>
|
||||||
|
#include <gio/gio.h>
|
||||||
|
|
||||||
|
struct _FpDeviceVirtualSdcp
|
||||||
|
{
|
||||||
|
FpSdcpDevice parent;
|
||||||
|
|
||||||
|
GSubprocess *proc;
|
||||||
|
GOutputStream *proc_stdin;
|
||||||
|
GInputStream *proc_stdout;
|
||||||
|
|
||||||
|
/* Only valid while a read/write is pending */
|
||||||
|
GByteArray *msg_out;
|
||||||
|
GByteArray *msg_in;
|
||||||
|
|
||||||
|
GByteArray *connect_msg;
|
||||||
|
};
|
||||||
|
|
||||||
|
G_DECLARE_FINAL_TYPE (FpDeviceVirtualSdcp, fpi_device_virtual_sdcp, FPI, DEVICE_VIRTUAL_SDCP, FpSdcpDevice)
|
||||||
|
G_DEFINE_TYPE (FpDeviceVirtualSdcp, fpi_device_virtual_sdcp, FP_TYPE_SDCP_DEVICE)
|
||||||
|
|
||||||
|
guint8 ca_1[] = {
|
||||||
|
0x30, 0x82, 0x03, 0xFD, 0x30, 0x82, 0x03, 0x82, 0xA0, 0x03, 0x02, 0x01, 0x02, 0x02, 0x13, 0x33,
|
||||||
|
0x00, 0x00, 0x00, 0x07, 0xE8, 0x9D, 0x61, 0x62, 0x4D, 0x46, 0x0F, 0x95, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x07, 0x30, 0x0A, 0x06, 0x08, 0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x04, 0x03, 0x02, 0x30, 0x81,
|
||||||
|
0x84, 0x31, 0x0B, 0x30, 0x09, 0x06, 0x03, 0x55, 0x04, 0x06, 0x13, 0x02, 0x55, 0x53, 0x31, 0x13,
|
||||||
|
0x30, 0x11, 0x06, 0x03, 0x55, 0x04, 0x08, 0x13, 0x0A, 0x57, 0x61, 0x73, 0x68, 0x69, 0x6E, 0x67,
|
||||||
|
0x74, 0x6F, 0x6E, 0x31, 0x10, 0x30, 0x0E, 0x06, 0x03, 0x55, 0x04, 0x07, 0x13, 0x07, 0x52, 0x65,
|
||||||
|
0x64, 0x6D, 0x6F, 0x6E, 0x64, 0x31, 0x1E, 0x30, 0x1C, 0x06, 0x03, 0x55, 0x04, 0x0A, 0x13, 0x15,
|
||||||
|
0x4D, 0x69, 0x63, 0x72, 0x6F, 0x73, 0x6F, 0x66, 0x74, 0x20, 0x43, 0x6F, 0x72, 0x70, 0x6F, 0x72,
|
||||||
|
0x61, 0x74, 0x69, 0x6F, 0x6E, 0x31, 0x2E, 0x30, 0x2C, 0x06, 0x03, 0x55, 0x04, 0x03, 0x13, 0x25,
|
||||||
|
0x57, 0x69, 0x6E, 0x64, 0x6F, 0x77, 0x73, 0x20, 0x48, 0x65, 0x6C, 0x6C, 0x6F, 0x20, 0x53, 0x65,
|
||||||
|
0x63, 0x75, 0x72, 0x65, 0x20, 0x44, 0x65, 0x76, 0x69, 0x63, 0x65, 0x73, 0x20, 0x50, 0x43, 0x41,
|
||||||
|
0x20, 0x32, 0x30, 0x31, 0x38, 0x30, 0x1E, 0x17, 0x0D, 0x31, 0x38, 0x30, 0x31, 0x33, 0x31, 0x31,
|
||||||
|
0x39, 0x35, 0x34, 0x35, 0x33, 0x5A, 0x17, 0x0D, 0x32, 0x38, 0x30, 0x31, 0x33, 0x31, 0x32, 0x30,
|
||||||
|
0x30, 0x34, 0x35, 0x33, 0x5A, 0x30, 0x7D, 0x31, 0x0B, 0x30, 0x09, 0x06, 0x03, 0x55, 0x04, 0x06,
|
||||||
|
0x13, 0x02, 0x55, 0x53, 0x31, 0x13, 0x30, 0x11, 0x06, 0x03, 0x55, 0x04, 0x08, 0x13, 0x0A, 0x57,
|
||||||
|
0x61, 0x73, 0x68, 0x69, 0x6E, 0x67, 0x74, 0x6F, 0x6E, 0x31, 0x10, 0x30, 0x0E, 0x06, 0x03, 0x55,
|
||||||
|
0x04, 0x07, 0x13, 0x07, 0x52, 0x65, 0x64, 0x6D, 0x6F, 0x6E, 0x64, 0x31, 0x1E, 0x30, 0x1C, 0x06,
|
||||||
|
0x03, 0x55, 0x04, 0x0A, 0x13, 0x15, 0x4D, 0x69, 0x63, 0x72, 0x6F, 0x73, 0x6F, 0x66, 0x74, 0x20,
|
||||||
|
0x43, 0x6F, 0x72, 0x70, 0x6F, 0x72, 0x61, 0x74, 0x69, 0x6F, 0x6E, 0x31, 0x27, 0x30, 0x25, 0x06,
|
||||||
|
0x03, 0x55, 0x04, 0x03, 0x13, 0x1E, 0x57, 0x69, 0x6E, 0x64, 0x6F, 0x77, 0x73, 0x20, 0x48, 0x65,
|
||||||
|
0x6C, 0x6C, 0x6F, 0x20, 0x31, 0x39, 0x42, 0x39, 0x32, 0x39, 0x36, 0x35, 0x20, 0x43, 0x41, 0x20,
|
||||||
|
0x32, 0x30, 0x31, 0x38, 0x30, 0x59, 0x30, 0x13, 0x06, 0x07, 0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x02,
|
||||||
|
0x01, 0x06, 0x08, 0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x03, 0x01, 0x07, 0x03, 0x42, 0x00, 0x04, 0xBE,
|
||||||
|
0x4B, 0x90, 0x6E, 0x24, 0xFC, 0xA1, 0x53, 0xC8, 0xA7, 0x3C, 0x70, 0xE8, 0x97, 0xCD, 0x1B, 0x31,
|
||||||
|
0xE4, 0x95, 0x91, 0x7A, 0x58, 0xA2, 0x86, 0xA8, 0x70, 0xF6, 0x09, 0x30, 0x77, 0x99, 0x3D, 0x10,
|
||||||
|
0xDF, 0xF7, 0x95, 0x0F, 0x68, 0x83, 0xE6, 0xA4, 0x11, 0x7C, 0xDA, 0x82, 0xE7, 0x0B, 0x8B, 0xF2,
|
||||||
|
0x9D, 0x6B, 0x5B, 0xF5, 0x3E, 0x77, 0xB4, 0xC1, 0x0E, 0x49, 0x00, 0x83, 0xBA, 0x94, 0xF8, 0xA3,
|
||||||
|
0x82, 0x01, 0xD7, 0x30, 0x82, 0x01, 0xD3, 0x30, 0x10, 0x06, 0x09, 0x2B, 0x06, 0x01, 0x04, 0x01,
|
||||||
|
0x82, 0x37, 0x15, 0x01, 0x04, 0x03, 0x02, 0x01, 0x00, 0x30, 0x1D, 0x06, 0x03, 0x55, 0x1D, 0x0E,
|
||||||
|
0x04, 0x16, 0x04, 0x14, 0x13, 0x93, 0xC8, 0xCD, 0xF2, 0x23, 0x9A, 0x2D, 0xC6, 0x9B, 0x2A, 0xEB,
|
||||||
|
0x9A, 0xAB, 0x99, 0x0B, 0x56, 0x04, 0x5E, 0x7C, 0x30, 0x65, 0x06, 0x03, 0x55, 0x1D, 0x20, 0x04,
|
||||||
|
0x5E, 0x30, 0x5C, 0x30, 0x06, 0x06, 0x04, 0x55, 0x1D, 0x20, 0x00, 0x30, 0x52, 0x06, 0x0C, 0x2B,
|
||||||
|
0x06, 0x01, 0x04, 0x01, 0x82, 0x37, 0x4C, 0x83, 0x7D, 0x01, 0x01, 0x30, 0x42, 0x30, 0x40, 0x06,
|
||||||
|
0x08, 0x2B, 0x06, 0x01, 0x05, 0x05, 0x07, 0x02, 0x01, 0x16, 0x34, 0x68, 0x74, 0x74, 0x70, 0x3A,
|
||||||
|
0x2F, 0x2F, 0x77, 0x77, 0x77, 0x2E, 0x6D, 0x69, 0x63, 0x72, 0x6F, 0x73, 0x6F, 0x66, 0x74, 0x2E,
|
||||||
|
0x63, 0x6F, 0x6D, 0x2F, 0x70, 0x6B, 0x69, 0x6F, 0x70, 0x73, 0x2F, 0x44, 0x6F, 0x63, 0x73, 0x2F,
|
||||||
|
0x52, 0x65, 0x70, 0x6F, 0x73, 0x69, 0x74, 0x6F, 0x72, 0x79, 0x2E, 0x68, 0x74, 0x6D, 0x00, 0x30,
|
||||||
|
0x19, 0x06, 0x09, 0x2B, 0x06, 0x01, 0x04, 0x01, 0x82, 0x37, 0x14, 0x02, 0x04, 0x0C, 0x1E, 0x0A,
|
||||||
|
0x00, 0x53, 0x00, 0x75, 0x00, 0x62, 0x00, 0x43, 0x00, 0x41, 0x30, 0x0B, 0x06, 0x03, 0x55, 0x1D,
|
||||||
|
0x0F, 0x04, 0x04, 0x03, 0x02, 0x01, 0x86, 0x30, 0x0F, 0x06, 0x03, 0x55, 0x1D, 0x13, 0x01, 0x01,
|
||||||
|
0xFF, 0x04, 0x05, 0x30, 0x03, 0x01, 0x01, 0xFF, 0x30, 0x1F, 0x06, 0x03, 0x55, 0x1D, 0x23, 0x04,
|
||||||
|
0x18, 0x30, 0x16, 0x80, 0x14, 0xDA, 0xCA, 0x4B, 0xD0, 0x4C, 0x56, 0x03, 0x27, 0x5F, 0x97, 0xEB,
|
||||||
|
0x75, 0xA3, 0x02, 0xC3, 0xBF, 0x45, 0x9C, 0xF8, 0xB1, 0x30, 0x68, 0x06, 0x03, 0x55, 0x1D, 0x1F,
|
||||||
|
0x04, 0x61, 0x30, 0x5F, 0x30, 0x5D, 0xA0, 0x5B, 0xA0, 0x59, 0x86, 0x57, 0x68, 0x74, 0x74, 0x70,
|
||||||
|
0x3A, 0x2F, 0x2F, 0x77, 0x77, 0x77, 0x2E, 0x6D, 0x69, 0x63, 0x72, 0x6F, 0x73, 0x6F, 0x66, 0x74,
|
||||||
|
0x2E, 0x63, 0x6F, 0x6D, 0x2F, 0x70, 0x6B, 0x69, 0x6F, 0x70, 0x73, 0x2F, 0x63, 0x72, 0x6C, 0x2F,
|
||||||
|
0x57, 0x69, 0x6E, 0x64, 0x6F, 0x77, 0x73, 0x25, 0x32, 0x30, 0x48, 0x65, 0x6C, 0x6C, 0x6F, 0x25,
|
||||||
|
0x32, 0x30, 0x53, 0x65, 0x63, 0x75, 0x72, 0x65, 0x25, 0x32, 0x30, 0x44, 0x65, 0x76, 0x69, 0x63,
|
||||||
|
0x65, 0x73, 0x25, 0x32, 0x30, 0x50, 0x43, 0x41, 0x25, 0x32, 0x30, 0x32, 0x30, 0x31, 0x38, 0x2E,
|
||||||
|
0x63, 0x72, 0x6C, 0x30, 0x75, 0x06, 0x08, 0x2B, 0x06, 0x01, 0x05, 0x05, 0x07, 0x01, 0x01, 0x04,
|
||||||
|
0x69, 0x30, 0x67, 0x30, 0x65, 0x06, 0x08, 0x2B, 0x06, 0x01, 0x05, 0x05, 0x07, 0x30, 0x02, 0x86,
|
||||||
|
0x59, 0x68, 0x74, 0x74, 0x70, 0x3A, 0x2F, 0x2F, 0x77, 0x77, 0x77, 0x2E, 0x6D, 0x69, 0x63, 0x72,
|
||||||
|
0x6F, 0x73, 0x6F, 0x66, 0x74, 0x2E, 0x63, 0x6F, 0x6D, 0x2F, 0x70, 0x6B, 0x69, 0x6F, 0x70, 0x73,
|
||||||
|
0x2F, 0x63, 0x65, 0x72, 0x74, 0x73, 0x2F, 0x57, 0x69, 0x6E, 0x64, 0x6F, 0x77, 0x73, 0x25, 0x32,
|
||||||
|
0x30, 0x48, 0x65, 0x6C, 0x6C, 0x6F, 0x25, 0x32, 0x30, 0x53, 0x65, 0x63, 0x75, 0x72, 0x65, 0x25,
|
||||||
|
0x32, 0x30, 0x44, 0x65, 0x76, 0x69, 0x63, 0x65, 0x73, 0x25, 0x32, 0x30, 0x50, 0x43, 0x41, 0x25,
|
||||||
|
0x32, 0x30, 0x32, 0x30, 0x31, 0x38, 0x2E, 0x63, 0x72, 0x74, 0x30, 0x0A, 0x06, 0x08, 0x2A, 0x86,
|
||||||
|
0x48, 0xCE, 0x3D, 0x04, 0x03, 0x02, 0x03, 0x69, 0x00, 0x30, 0x66, 0x02, 0x31, 0x00, 0x87, 0xB6,
|
||||||
|
0x82, 0xF3, 0xDA, 0xBE, 0xB1, 0x7B, 0x98, 0x7D, 0x3D, 0x0A, 0x90, 0xA8, 0xF5, 0xBF, 0x15, 0xC3,
|
||||||
|
0xEE, 0x8A, 0x4E, 0xC0, 0x7B, 0x10, 0x1D, 0xA9, 0xE3, 0x0B, 0xEC, 0x2C, 0x53, 0x4E, 0xA7, 0xBD,
|
||||||
|
0xF1, 0x6C, 0xAD, 0x18, 0x55, 0xBA, 0x25, 0x73, 0x55, 0xB7, 0x5B, 0x12, 0x24, 0xF4, 0x02, 0x31,
|
||||||
|
0x00, 0xAF, 0x02, 0x9C, 0x4B, 0x92, 0xD0, 0x72, 0xA5, 0x80, 0xCA, 0x69, 0x2B, 0x38, 0x50, 0x64,
|
||||||
|
0xD8, 0x58, 0x9E, 0xEA, 0xD6, 0x35, 0xCF, 0x68, 0x98, 0x92, 0x81, 0x09, 0x61, 0xC2, 0xBD, 0xB1,
|
||||||
|
0x4C, 0x7F, 0xAE, 0x55, 0x7B, 0xFC, 0x22, 0xDD, 0xD6, 0xB7, 0x7C, 0xB5, 0xA8, 0x18, 0x5D, 0x33,
|
||||||
|
0x04
|
||||||
|
};
|
||||||
|
guint8 ca_2[] = {
|
||||||
|
0x30, 0x82, 0x04, 0x56, 0x30, 0x82, 0x03, 0xDC, 0xA0, 0x03, 0x02, 0x01, 0x02, 0x02, 0x13, 0x33,
|
||||||
|
0x00, 0x00, 0x00, 0x03, 0x6C, 0xCF, 0xED, 0xE2, 0x44, 0x70, 0x19, 0xBF, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x03, 0x30, 0x0A, 0x06, 0x08, 0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x04, 0x03, 0x03, 0x30, 0x81,
|
||||||
|
0x94, 0x31, 0x0B, 0x30, 0x09, 0x06, 0x03, 0x55, 0x04, 0x06, 0x13, 0x02, 0x55, 0x53, 0x31, 0x13,
|
||||||
|
0x30, 0x11, 0x06, 0x03, 0x55, 0x04, 0x08, 0x13, 0x0A, 0x57, 0x61, 0x73, 0x68, 0x69, 0x6E, 0x67,
|
||||||
|
0x74, 0x6F, 0x6E, 0x31, 0x10, 0x30, 0x0E, 0x06, 0x03, 0x55, 0x04, 0x07, 0x13, 0x07, 0x52, 0x65,
|
||||||
|
0x64, 0x6D, 0x6F, 0x6E, 0x64, 0x31, 0x1E, 0x30, 0x1C, 0x06, 0x03, 0x55, 0x04, 0x0A, 0x13, 0x15,
|
||||||
|
0x4D, 0x69, 0x63, 0x72, 0x6F, 0x73, 0x6F, 0x66, 0x74, 0x20, 0x43, 0x6F, 0x72, 0x70, 0x6F, 0x72,
|
||||||
|
0x61, 0x74, 0x69, 0x6F, 0x6E, 0x31, 0x3E, 0x30, 0x3C, 0x06, 0x03, 0x55, 0x04, 0x03, 0x13, 0x35,
|
||||||
|
0x4D, 0x69, 0x63, 0x72, 0x6F, 0x73, 0x6F, 0x66, 0x74, 0x20, 0x45, 0x43, 0x43, 0x20, 0x44, 0x65,
|
||||||
|
0x76, 0x69, 0x63, 0x65, 0x73, 0x20, 0x52, 0x6F, 0x6F, 0x74, 0x20, 0x43, 0x65, 0x72, 0x74, 0x69,
|
||||||
|
0x66, 0x69, 0x63, 0x61, 0x74, 0x65, 0x20, 0x41, 0x75, 0x74, 0x68, 0x6F, 0x72, 0x69, 0x74, 0x79,
|
||||||
|
0x20, 0x32, 0x30, 0x31, 0x37, 0x30, 0x1E, 0x17, 0x0D, 0x31, 0x38, 0x30, 0x31, 0x32, 0x35, 0x31,
|
||||||
|
0x39, 0x34, 0x39, 0x33, 0x38, 0x5A, 0x17, 0x0D, 0x33, 0x33, 0x30, 0x31, 0x32, 0x35, 0x31, 0x39,
|
||||||
|
0x35, 0x39, 0x33, 0x38, 0x5A, 0x30, 0x81, 0x84, 0x31, 0x0B, 0x30, 0x09, 0x06, 0x03, 0x55, 0x04,
|
||||||
|
0x06, 0x13, 0x02, 0x55, 0x53, 0x31, 0x13, 0x30, 0x11, 0x06, 0x03, 0x55, 0x04, 0x08, 0x13, 0x0A,
|
||||||
|
0x57, 0x61, 0x73, 0x68, 0x69, 0x6E, 0x67, 0x74, 0x6F, 0x6E, 0x31, 0x10, 0x30, 0x0E, 0x06, 0x03,
|
||||||
|
0x55, 0x04, 0x07, 0x13, 0x07, 0x52, 0x65, 0x64, 0x6D, 0x6F, 0x6E, 0x64, 0x31, 0x1E, 0x30, 0x1C,
|
||||||
|
0x06, 0x03, 0x55, 0x04, 0x0A, 0x13, 0x15, 0x4D, 0x69, 0x63, 0x72, 0x6F, 0x73, 0x6F, 0x66, 0x74,
|
||||||
|
0x20, 0x43, 0x6F, 0x72, 0x70, 0x6F, 0x72, 0x61, 0x74, 0x69, 0x6F, 0x6E, 0x31, 0x2E, 0x30, 0x2C,
|
||||||
|
0x06, 0x03, 0x55, 0x04, 0x03, 0x13, 0x25, 0x57, 0x69, 0x6E, 0x64, 0x6F, 0x77, 0x73, 0x20, 0x48,
|
||||||
|
0x65, 0x6C, 0x6C, 0x6F, 0x20, 0x53, 0x65, 0x63, 0x75, 0x72, 0x65, 0x20, 0x44, 0x65, 0x76, 0x69,
|
||||||
|
0x63, 0x65, 0x73, 0x20, 0x50, 0x43, 0x41, 0x20, 0x32, 0x30, 0x31, 0x38, 0x30, 0x76, 0x30, 0x10,
|
||||||
|
0x06, 0x07, 0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x02, 0x01, 0x06, 0x05, 0x2B, 0x81, 0x04, 0x00, 0x22,
|
||||||
|
0x03, 0x62, 0x00, 0x04, 0x1D, 0xDD, 0x08, 0x02, 0x03, 0x25, 0x75, 0x20, 0xE2, 0x71, 0x8B, 0xAD,
|
||||||
|
0x28, 0x09, 0x82, 0xE9, 0x06, 0xEE, 0x83, 0xC5, 0x3A, 0x6C, 0x4B, 0x71, 0x92, 0x50, 0x4E, 0x20,
|
||||||
|
0xE9, 0x72, 0xB4, 0xFC, 0x53, 0x2A, 0xEF, 0x5D, 0xCC, 0x9A, 0xB4, 0xCD, 0x76, 0xB8, 0x94, 0x97,
|
||||||
|
0x44, 0xB2, 0x71, 0x0E, 0xC9, 0xB1, 0x16, 0x03, 0xA1, 0x65, 0x2B, 0xB9, 0xE8, 0x5D, 0x5F, 0xF2,
|
||||||
|
0x30, 0x2E, 0xDD, 0xB1, 0x2B, 0x20, 0xFC, 0xBE, 0x00, 0x88, 0xEA, 0x1F, 0xA7, 0x7F, 0x99, 0x84,
|
||||||
|
0x98, 0x7C, 0x71, 0x3E, 0x4D, 0x34, 0x83, 0x69, 0x9B, 0x08, 0xCB, 0x78, 0xB2, 0x4B, 0xBD, 0xD7,
|
||||||
|
0x3E, 0xBE, 0x67, 0xA0, 0xA3, 0x82, 0x01, 0xFC, 0x30, 0x82, 0x01, 0xF8, 0x30, 0x10, 0x06, 0x09,
|
||||||
|
0x2B, 0x06, 0x01, 0x04, 0x01, 0x82, 0x37, 0x15, 0x01, 0x04, 0x03, 0x02, 0x01, 0x00, 0x30, 0x1D,
|
||||||
|
0x06, 0x03, 0x55, 0x1D, 0x0E, 0x04, 0x16, 0x04, 0x14, 0xDA, 0xCA, 0x4B, 0xD0, 0x4C, 0x56, 0x03,
|
||||||
|
0x27, 0x5F, 0x97, 0xEB, 0x75, 0xA3, 0x02, 0xC3, 0xBF, 0x45, 0x9C, 0xF8, 0xB1, 0x30, 0x65, 0x06,
|
||||||
|
0x03, 0x55, 0x1D, 0x20, 0x04, 0x5E, 0x30, 0x5C, 0x30, 0x06, 0x06, 0x04, 0x55, 0x1D, 0x20, 0x00,
|
||||||
|
0x30, 0x52, 0x06, 0x0C, 0x2B, 0x06, 0x01, 0x04, 0x01, 0x82, 0x37, 0x4C, 0x83, 0x7D, 0x01, 0x01,
|
||||||
|
0x30, 0x42, 0x30, 0x40, 0x06, 0x08, 0x2B, 0x06, 0x01, 0x05, 0x05, 0x07, 0x02, 0x01, 0x16, 0x34,
|
||||||
|
0x68, 0x74, 0x74, 0x70, 0x3A, 0x2F, 0x2F, 0x77, 0x77, 0x77, 0x2E, 0x6D, 0x69, 0x63, 0x72, 0x6F,
|
||||||
|
0x73, 0x6F, 0x66, 0x74, 0x2E, 0x63, 0x6F, 0x6D, 0x2F, 0x70, 0x6B, 0x69, 0x6F, 0x70, 0x73, 0x2F,
|
||||||
|
0x44, 0x6F, 0x63, 0x73, 0x2F, 0x52, 0x65, 0x70, 0x6F, 0x73, 0x69, 0x74, 0x6F, 0x72, 0x79, 0x2E,
|
||||||
|
0x68, 0x74, 0x6D, 0x00, 0x30, 0x19, 0x06, 0x09, 0x2B, 0x06, 0x01, 0x04, 0x01, 0x82, 0x37, 0x14,
|
||||||
|
0x02, 0x04, 0x0C, 0x1E, 0x0A, 0x00, 0x53, 0x00, 0x75, 0x00, 0x62, 0x00, 0x43, 0x00, 0x41, 0x30,
|
||||||
|
0x0B, 0x06, 0x03, 0x55, 0x1D, 0x0F, 0x04, 0x04, 0x03, 0x02, 0x01, 0x86, 0x30, 0x0F, 0x06, 0x03,
|
||||||
|
0x55, 0x1D, 0x13, 0x01, 0x01, 0xFF, 0x04, 0x05, 0x30, 0x03, 0x01, 0x01, 0xFF, 0x30, 0x1F, 0x06,
|
||||||
|
0x03, 0x55, 0x1D, 0x23, 0x04, 0x18, 0x30, 0x16, 0x80, 0x14, 0x14, 0xDA, 0x5B, 0xF1, 0x0E, 0x66,
|
||||||
|
0x47, 0xD1, 0x5D, 0x13, 0x5F, 0x5B, 0x7A, 0xEB, 0xEB, 0x5F, 0x01, 0x08, 0xB5, 0x49, 0x30, 0x7A,
|
||||||
|
0x06, 0x03, 0x55, 0x1D, 0x1F, 0x04, 0x73, 0x30, 0x71, 0x30, 0x6F, 0xA0, 0x6D, 0xA0, 0x6B, 0x86,
|
||||||
|
0x69, 0x68, 0x74, 0x74, 0x70, 0x3A, 0x2F, 0x2F, 0x77, 0x77, 0x77, 0x2E, 0x6D, 0x69, 0x63, 0x72,
|
||||||
|
0x6F, 0x73, 0x6F, 0x66, 0x74, 0x2E, 0x63, 0x6F, 0x6D, 0x2F, 0x70, 0x6B, 0x69, 0x6F, 0x70, 0x73,
|
||||||
|
0x2F, 0x63, 0x72, 0x6C, 0x2F, 0x4D, 0x69, 0x63, 0x72, 0x6F, 0x73, 0x6F, 0x66, 0x74, 0x25, 0x32,
|
||||||
|
0x30, 0x45, 0x43, 0x43, 0x25, 0x32, 0x30, 0x44, 0x65, 0x76, 0x69, 0x63, 0x65, 0x73, 0x25, 0x32,
|
||||||
|
0x30, 0x52, 0x6F, 0x6F, 0x74, 0x25, 0x32, 0x30, 0x43, 0x65, 0x72, 0x74, 0x69, 0x66, 0x69, 0x63,
|
||||||
|
0x61, 0x74, 0x65, 0x25, 0x32, 0x30, 0x41, 0x75, 0x74, 0x68, 0x6F, 0x72, 0x69, 0x74, 0x79, 0x25,
|
||||||
|
0x32, 0x30, 0x32, 0x30, 0x31, 0x37, 0x2E, 0x63, 0x72, 0x6C, 0x30, 0x81, 0x87, 0x06, 0x08, 0x2B,
|
||||||
|
0x06, 0x01, 0x05, 0x05, 0x07, 0x01, 0x01, 0x04, 0x7B, 0x30, 0x79, 0x30, 0x77, 0x06, 0x08, 0x2B,
|
||||||
|
0x06, 0x01, 0x05, 0x05, 0x07, 0x30, 0x02, 0x86, 0x6B, 0x68, 0x74, 0x74, 0x70, 0x3A, 0x2F, 0x2F,
|
||||||
|
0x77, 0x77, 0x77, 0x2E, 0x6D, 0x69, 0x63, 0x72, 0x6F, 0x73, 0x6F, 0x66, 0x74, 0x2E, 0x63, 0x6F,
|
||||||
|
0x6D, 0x2F, 0x70, 0x6B, 0x69, 0x6F, 0x70, 0x73, 0x2F, 0x63, 0x65, 0x72, 0x74, 0x73, 0x2F, 0x4D,
|
||||||
|
0x69, 0x63, 0x72, 0x6F, 0x73, 0x6F, 0x66, 0x74, 0x25, 0x32, 0x30, 0x45, 0x43, 0x43, 0x25, 0x32,
|
||||||
|
0x30, 0x44, 0x65, 0x76, 0x69, 0x63, 0x65, 0x73, 0x25, 0x32, 0x30, 0x52, 0x6F, 0x6F, 0x74, 0x25,
|
||||||
|
0x32, 0x30, 0x43, 0x65, 0x72, 0x74, 0x69, 0x66, 0x69, 0x63, 0x61, 0x74, 0x65, 0x25, 0x32, 0x30,
|
||||||
|
0x41, 0x75, 0x74, 0x68, 0x6F, 0x72, 0x69, 0x74, 0x79, 0x25, 0x32, 0x30, 0x32, 0x30, 0x31, 0x37,
|
||||||
|
0x2E, 0x63, 0x72, 0x74, 0x30, 0x0A, 0x06, 0x08, 0x2A, 0x86, 0x48, 0xCE, 0x3D, 0x04, 0x03, 0x03,
|
||||||
|
0x03, 0x68, 0x00, 0x30, 0x65, 0x02, 0x30, 0x56, 0x2A, 0xAD, 0x72, 0x4C, 0xB9, 0x8C, 0xB3, 0x23,
|
||||||
|
0x80, 0xF5, 0x5F, 0xF8, 0x21, 0x94, 0x66, 0x0F, 0x76, 0x77, 0xE2, 0x7B, 0x03, 0xDD, 0x30, 0x5E,
|
||||||
|
0xCB, 0x90, 0xCA, 0x78, 0xE6, 0x0B, 0x2D, 0x12, 0xE5, 0xF7, 0x67, 0x31, 0x58, 0x71, 0xE6, 0xF3,
|
||||||
|
0x64, 0xC1, 0x04, 0xB3, 0x8B, 0xE9, 0xE2, 0x02, 0x31, 0x00, 0xB9, 0x20, 0x61, 0xB9, 0xD0, 0x5E,
|
||||||
|
0x3A, 0xA4, 0xA2, 0x8A, 0xFE, 0x1D, 0xFC, 0x27, 0x61, 0x0B, 0x98, 0x16, 0x8C, 0x02, 0x9C, 0x20,
|
||||||
|
0x7F, 0xEE, 0xF3, 0xCB, 0x1F, 0x0A, 0x37, 0x62, 0xB1, 0x8E, 0xCE, 0xD9, 0x9A, 0x9E, 0xAC, 0xE6,
|
||||||
|
0x1A, 0xD4, 0xB8, 0xF1, 0xA8, 0x2B, 0xB1, 0xB4, 0x40, 0x9B
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
static void
|
||||||
|
msg_written_cb (GObject *source_object,
|
||||||
|
GAsyncResult *res,
|
||||||
|
gpointer user_data)
|
||||||
|
{
|
||||||
|
GError *error = NULL;
|
||||||
|
GOutputStream *stream = G_OUTPUT_STREAM (source_object);
|
||||||
|
FpiSsm *ssm = user_data;
|
||||||
|
FpDeviceVirtualSdcp *self = FPI_DEVICE_VIRTUAL_SDCP (fpi_ssm_get_device (ssm));
|
||||||
|
|
||||||
|
g_clear_pointer (&self->msg_out, g_byte_array_unref);
|
||||||
|
g_assert (self->msg_out == NULL);
|
||||||
|
|
||||||
|
if (!g_output_stream_write_all_finish (stream, res, NULL, &error))
|
||||||
|
{
|
||||||
|
fpi_ssm_mark_failed (ssm, error);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
fpi_ssm_next_state (ssm);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
msg_received_cb (GObject *source_object,
|
||||||
|
GAsyncResult *res,
|
||||||
|
gpointer user_data)
|
||||||
|
{
|
||||||
|
GError *error = NULL;
|
||||||
|
GInputStream *stream = G_INPUT_STREAM (source_object);
|
||||||
|
FpiSsm *ssm = user_data;
|
||||||
|
FpDeviceVirtualSdcp *self = FPI_DEVICE_VIRTUAL_SDCP (fpi_ssm_get_device (ssm));
|
||||||
|
gsize read;
|
||||||
|
|
||||||
|
g_assert (self->msg_out == NULL);
|
||||||
|
|
||||||
|
if (!g_input_stream_read_all_finish (stream, res, &read, &error) ||
|
||||||
|
read != self->msg_in->len)
|
||||||
|
{
|
||||||
|
g_clear_pointer (&self->msg_in, g_byte_array_unref);
|
||||||
|
|
||||||
|
if (!error)
|
||||||
|
error = fpi_device_error_new_msg (FP_DEVICE_ERROR_PROTO,
|
||||||
|
"Received EOF while reading from test binary.");
|
||||||
|
|
||||||
|
fpi_ssm_mark_failed (ssm, error);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
fpi_ssm_next_state (ssm);
|
||||||
|
}
|
||||||
|
|
||||||
|
enum {
|
||||||
|
SEND_MESSAGE,
|
||||||
|
RECV_MESSAGE,
|
||||||
|
SEND_RECV_STATES
|
||||||
|
};
|
||||||
|
|
||||||
|
static void
|
||||||
|
send_recv_ssm (FpiSsm *ssm, FpDevice *dev)
|
||||||
|
{
|
||||||
|
FpDeviceVirtualSdcp *self = FPI_DEVICE_VIRTUAL_SDCP (dev);
|
||||||
|
|
||||||
|
switch (fpi_ssm_get_cur_state (ssm))
|
||||||
|
{
|
||||||
|
case SEND_MESSAGE:
|
||||||
|
g_output_stream_write_all_async (self->proc_stdin,
|
||||||
|
self->msg_out->data,
|
||||||
|
self->msg_out->len,
|
||||||
|
G_PRIORITY_DEFAULT,
|
||||||
|
fpi_device_get_cancellable (FP_DEVICE (dev)),
|
||||||
|
msg_written_cb,
|
||||||
|
ssm);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case RECV_MESSAGE:
|
||||||
|
g_input_stream_read_all_async (self->proc_stdout,
|
||||||
|
self->msg_in->data,
|
||||||
|
self->msg_in->len,
|
||||||
|
G_PRIORITY_DEFAULT,
|
||||||
|
fpi_device_get_cancellable (FP_DEVICE (dev)),
|
||||||
|
msg_received_cb,
|
||||||
|
ssm);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
connect_2_cb (FpiSsm *ssm, FpDevice *dev, GError *error)
|
||||||
|
{
|
||||||
|
g_autoptr(GBytes) recv_data = NULL;
|
||||||
|
g_autoptr(GBytes) r_d = NULL;
|
||||||
|
g_autoptr(FpiSdcpClaim) claim = NULL;
|
||||||
|
g_autoptr(GBytes) mac = NULL;
|
||||||
|
g_autoptr(GBytes) ca_1_bytes = NULL, ca_2_bytes = NULL;
|
||||||
|
FpDeviceVirtualSdcp *self = FPI_DEVICE_VIRTUAL_SDCP (dev);
|
||||||
|
guint16 cert_size;
|
||||||
|
|
||||||
|
if (error)
|
||||||
|
{
|
||||||
|
fpi_sdcp_device_connect_complete (FP_SDCP_DEVICE (dev), NULL, NULL, NULL, error);
|
||||||
|
g_clear_pointer (&self->connect_msg, g_byte_array_unref);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy (&cert_size, self->connect_msg->data + 32, 2);
|
||||||
|
g_byte_array_append (self->connect_msg, self->msg_in->data, self->msg_in->len);
|
||||||
|
g_clear_pointer (&self->msg_in, g_byte_array_unref);
|
||||||
|
/* Double check that the size is correct. */
|
||||||
|
g_assert (self->connect_msg->len == 32 + (2 + cert_size + 65 + 65 + 32 + 64 + 64) + 32);
|
||||||
|
recv_data = g_byte_array_free_to_bytes (g_steal_pointer (&self->connect_msg));
|
||||||
|
|
||||||
|
claim = fpi_sdcp_claim_new ();
|
||||||
|
r_d = g_bytes_new_from_bytes (recv_data, 0, 32);
|
||||||
|
claim->cert_m = g_bytes_new_from_bytes (recv_data, 34, cert_size);
|
||||||
|
claim->pk_d = g_bytes_new_from_bytes (recv_data, 34 + cert_size, 65);
|
||||||
|
claim->pk_f = g_bytes_new_from_bytes (recv_data, 34 + cert_size + 65, 65);
|
||||||
|
claim->h_f = g_bytes_new_from_bytes (recv_data, 34 + cert_size + 65 + 65, 32);
|
||||||
|
claim->s_m = g_bytes_new_from_bytes (recv_data, 34 + cert_size + 65 + 65 + 32, 64);
|
||||||
|
claim->s_d = g_bytes_new_from_bytes (recv_data, 34 + cert_size + 65 + 65 + 32 + 64, 64);
|
||||||
|
mac = g_bytes_new_from_bytes (recv_data, 34 + cert_size + 65 + 65 + 32 + 64 + 64, 32);
|
||||||
|
|
||||||
|
ca_1_bytes = g_bytes_new_static (ca_1, G_N_ELEMENTS (ca_1));
|
||||||
|
ca_2_bytes = g_bytes_new_static (ca_2, G_N_ELEMENTS (ca_2));
|
||||||
|
|
||||||
|
fpi_sdcp_device_set_intermediat_cas (FP_SDCP_DEVICE (dev),
|
||||||
|
ca_1_bytes,
|
||||||
|
ca_2_bytes);
|
||||||
|
|
||||||
|
fpi_sdcp_device_connect_complete (FP_SDCP_DEVICE (dev), r_d, claim, mac, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
connect_1_cb (FpiSsm *ssm, FpDevice *dev, GError *error)
|
||||||
|
{
|
||||||
|
FpDeviceVirtualSdcp *self = FPI_DEVICE_VIRTUAL_SDCP (dev);
|
||||||
|
guint16 cert_size;
|
||||||
|
|
||||||
|
if (error)
|
||||||
|
{
|
||||||
|
fpi_sdcp_device_connect_complete (FP_SDCP_DEVICE (dev), NULL, NULL, NULL, error);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
g_clear_pointer (&self->connect_msg, g_byte_array_unref);
|
||||||
|
self->connect_msg = g_steal_pointer (&self->msg_in);
|
||||||
|
|
||||||
|
memcpy (&cert_size, self->connect_msg->data + 32, 2);
|
||||||
|
|
||||||
|
/* Nothing to send and the rest to receive. */
|
||||||
|
self->msg_out = g_byte_array_new ();
|
||||||
|
self->msg_in = g_byte_array_new ();
|
||||||
|
g_byte_array_set_size (self->msg_in, 32 + (2 + cert_size + 65 + 65 + 32 + 64 + 64) + 32 - self->connect_msg->len);
|
||||||
|
|
||||||
|
/* New SSM */
|
||||||
|
ssm = fpi_ssm_new_full (FP_DEVICE (dev), send_recv_ssm, SEND_RECV_STATES, SEND_RECV_STATES, "connect 2");
|
||||||
|
fpi_ssm_start (ssm, connect_2_cb);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
connect (FpSdcpDevice *dev)
|
||||||
|
{
|
||||||
|
g_autoptr(GBytes) r_h = NULL;
|
||||||
|
g_autoptr(GBytes) pk_h = NULL;
|
||||||
|
FpDeviceVirtualSdcp *self = FPI_DEVICE_VIRTUAL_SDCP (dev);
|
||||||
|
FpiSsm *ssm;
|
||||||
|
|
||||||
|
G_DEBUG_HERE ();
|
||||||
|
|
||||||
|
g_assert (self->proc);
|
||||||
|
g_assert (self->connect_msg == NULL);
|
||||||
|
|
||||||
|
fpi_sdcp_device_get_connect_data (dev, &r_h, &pk_h);
|
||||||
|
|
||||||
|
self->msg_out = g_byte_array_new ();
|
||||||
|
g_byte_array_append (self->msg_out, (const guint8 *) "C", 1);
|
||||||
|
g_byte_array_append (self->msg_out,
|
||||||
|
g_bytes_get_data (r_h, NULL),
|
||||||
|
g_bytes_get_size (r_h));
|
||||||
|
g_byte_array_append (self->msg_out,
|
||||||
|
g_bytes_get_data (pk_h, NULL),
|
||||||
|
g_bytes_get_size (pk_h));
|
||||||
|
|
||||||
|
self->msg_in = g_byte_array_new ();
|
||||||
|
g_byte_array_set_size (self->msg_in, 34);
|
||||||
|
|
||||||
|
ssm = fpi_ssm_new_full (FP_DEVICE (dev), send_recv_ssm, SEND_RECV_STATES, SEND_RECV_STATES, "connect");
|
||||||
|
fpi_ssm_start (ssm, connect_1_cb);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
reconnect_cb (FpiSsm *ssm, FpDevice *dev, GError *error)
|
||||||
|
{
|
||||||
|
g_autoptr(GBytes) mac = NULL;
|
||||||
|
FpDeviceVirtualSdcp *self = FPI_DEVICE_VIRTUAL_SDCP (dev);
|
||||||
|
|
||||||
|
if (error)
|
||||||
|
{
|
||||||
|
fpi_sdcp_device_reconnect_complete (FP_SDCP_DEVICE (dev), mac, error);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
mac = g_byte_array_free_to_bytes (g_steal_pointer (&self->msg_in));
|
||||||
|
|
||||||
|
fpi_sdcp_device_reconnect_complete (FP_SDCP_DEVICE (dev), mac, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
reconnect (FpSdcpDevice *dev)
|
||||||
|
{
|
||||||
|
g_autoptr(GBytes) r_h = NULL;
|
||||||
|
FpDeviceVirtualSdcp *self = FPI_DEVICE_VIRTUAL_SDCP (dev);
|
||||||
|
FpiSsm *ssm;
|
||||||
|
|
||||||
|
G_DEBUG_HERE ();
|
||||||
|
|
||||||
|
g_assert (self->proc);
|
||||||
|
|
||||||
|
fpi_sdcp_device_get_reconnect_data (dev, &r_h);
|
||||||
|
|
||||||
|
self->msg_out = g_byte_array_new ();
|
||||||
|
g_byte_array_append (self->msg_out, (const guint8 *) "R", 1);
|
||||||
|
g_byte_array_append (self->msg_out,
|
||||||
|
g_bytes_get_data (r_h, NULL),
|
||||||
|
g_bytes_get_size (r_h));
|
||||||
|
|
||||||
|
self->msg_in = g_byte_array_new ();
|
||||||
|
g_byte_array_set_size (self->msg_in, 32);
|
||||||
|
|
||||||
|
ssm = fpi_ssm_new_full (FP_DEVICE (dev), send_recv_ssm, SEND_RECV_STATES, SEND_RECV_STATES, "connect 2");
|
||||||
|
fpi_ssm_start (ssm, reconnect_cb);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
enroll_begin_cb (FpiSsm *ssm, FpDevice *dev, GError *error)
|
||||||
|
{
|
||||||
|
g_autoptr(GBytes) nonce = NULL;
|
||||||
|
FpDeviceVirtualSdcp *self = FPI_DEVICE_VIRTUAL_SDCP (dev);
|
||||||
|
|
||||||
|
if (error)
|
||||||
|
{
|
||||||
|
fpi_sdcp_device_enroll_ready (FP_SDCP_DEVICE (dev), error);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
nonce = g_byte_array_free_to_bytes (g_steal_pointer (&self->msg_in));
|
||||||
|
|
||||||
|
fpi_sdcp_device_enroll_set_nonce (FP_SDCP_DEVICE (dev), nonce);
|
||||||
|
|
||||||
|
/* Claim that we completed one enroll step. */
|
||||||
|
fpi_device_enroll_progress (dev, 1, NULL, NULL);
|
||||||
|
|
||||||
|
/* And signal that we are ready to commit. */
|
||||||
|
fpi_sdcp_device_enroll_ready (FP_SDCP_DEVICE (dev), NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
enroll_begin (FpSdcpDevice *dev)
|
||||||
|
{
|
||||||
|
g_autoptr(GBytes) r_h = NULL;
|
||||||
|
FpDeviceVirtualSdcp *self = FPI_DEVICE_VIRTUAL_SDCP (dev);
|
||||||
|
FpiSsm *ssm;
|
||||||
|
|
||||||
|
G_DEBUG_HERE ();
|
||||||
|
|
||||||
|
g_assert (self->proc);
|
||||||
|
|
||||||
|
fpi_sdcp_device_get_reconnect_data (dev, &r_h);
|
||||||
|
|
||||||
|
self->msg_out = g_byte_array_new ();
|
||||||
|
g_byte_array_append (self->msg_out, (const guint8 *) "E", 1);
|
||||||
|
|
||||||
|
/* Expect 32 byte nonce */
|
||||||
|
self->msg_in = g_byte_array_new ();
|
||||||
|
g_byte_array_set_size (self->msg_in, 32);
|
||||||
|
|
||||||
|
ssm = fpi_ssm_new_full (FP_DEVICE (dev), send_recv_ssm, SEND_RECV_STATES, SEND_RECV_STATES, "enroll_begin");
|
||||||
|
fpi_ssm_start (ssm, enroll_begin_cb);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
enroll_commit_cb (FpiSsm *ssm, FpDevice *dev, GError *error)
|
||||||
|
{
|
||||||
|
FpDeviceVirtualSdcp *self = FPI_DEVICE_VIRTUAL_SDCP (dev);
|
||||||
|
|
||||||
|
g_clear_pointer (&self->msg_in, g_byte_array_unref);
|
||||||
|
|
||||||
|
if (error)
|
||||||
|
{
|
||||||
|
fpi_sdcp_device_enroll_ready (FP_SDCP_DEVICE (dev), error);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Signal that we have committed. We don't expect a response
|
||||||
|
* from the virtual device (even though that is kind of broken).
|
||||||
|
*/
|
||||||
|
fpi_sdcp_device_enroll_commit_complete (FP_SDCP_DEVICE (dev), NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
enroll_commit (FpSdcpDevice *dev, GBytes *id_in)
|
||||||
|
{
|
||||||
|
g_autoptr(GBytes) r_h = NULL;
|
||||||
|
g_autoptr(GBytes) id = id_in;
|
||||||
|
FpDeviceVirtualSdcp *self = FPI_DEVICE_VIRTUAL_SDCP (dev);
|
||||||
|
FpiSsm *ssm;
|
||||||
|
|
||||||
|
G_DEBUG_HERE ();
|
||||||
|
|
||||||
|
g_assert (self->proc);
|
||||||
|
|
||||||
|
fpi_sdcp_device_get_reconnect_data (dev, &r_h);
|
||||||
|
|
||||||
|
self->msg_out = g_byte_array_new ();
|
||||||
|
self->msg_in = g_byte_array_new ();
|
||||||
|
if (id)
|
||||||
|
{
|
||||||
|
g_byte_array_append (self->msg_out, (const guint8 *) "F", 1);
|
||||||
|
g_byte_array_append (self->msg_out,
|
||||||
|
g_bytes_get_data (id, NULL),
|
||||||
|
g_bytes_get_size (id));
|
||||||
|
|
||||||
|
/* NOTE: No response from device, assume commit works. */
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* Cancel enroll (does not receive a reply) */
|
||||||
|
g_byte_array_append (self->msg_out, (const guint8 *) "G", 1);
|
||||||
|
|
||||||
|
/* NOTE: No response from device, assume cancellation works. */
|
||||||
|
}
|
||||||
|
|
||||||
|
ssm = fpi_ssm_new_full (FP_DEVICE (dev), send_recv_ssm, SEND_RECV_STATES, SEND_RECV_STATES, "enroll_commit");
|
||||||
|
fpi_ssm_start (ssm, enroll_commit_cb);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
identify_cb (FpiSsm *ssm, FpDevice *dev, GError *error)
|
||||||
|
{
|
||||||
|
g_autoptr(GBytes) reply = NULL;
|
||||||
|
g_autoptr(GBytes) id = NULL;
|
||||||
|
g_autoptr(GBytes) mac = NULL;
|
||||||
|
FpDeviceVirtualSdcp *self = FPI_DEVICE_VIRTUAL_SDCP (dev);
|
||||||
|
|
||||||
|
if (error)
|
||||||
|
{
|
||||||
|
fpi_sdcp_device_identify_complete (FP_SDCP_DEVICE (dev), NULL, NULL, error);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
reply = g_byte_array_free_to_bytes (g_steal_pointer (&self->msg_in));
|
||||||
|
id = g_bytes_new_from_bytes (reply, 0, 32);
|
||||||
|
mac = g_bytes_new_from_bytes (reply, 32, 32);
|
||||||
|
|
||||||
|
fpi_sdcp_device_identify_complete (FP_SDCP_DEVICE (self), id, mac, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
identify (FpSdcpDevice *dev)
|
||||||
|
{
|
||||||
|
g_autoptr(GBytes) nonce = NULL;
|
||||||
|
FpDeviceVirtualSdcp *self = FPI_DEVICE_VIRTUAL_SDCP (dev);
|
||||||
|
FpiSsm *ssm;
|
||||||
|
|
||||||
|
G_DEBUG_HERE ();
|
||||||
|
|
||||||
|
g_assert (self->proc);
|
||||||
|
|
||||||
|
fpi_sdcp_device_get_identify_data (dev, &nonce);
|
||||||
|
|
||||||
|
self->msg_out = g_byte_array_new ();
|
||||||
|
g_byte_array_append (self->msg_out, (const guint8 *) "I", 1);
|
||||||
|
g_byte_array_append (self->msg_out, g_bytes_get_data (nonce, NULL), g_bytes_get_size (nonce));
|
||||||
|
|
||||||
|
/* Expect 64 byte nonce */
|
||||||
|
self->msg_in = g_byte_array_new ();
|
||||||
|
g_byte_array_set_size (self->msg_in, 64);
|
||||||
|
|
||||||
|
ssm = fpi_ssm_new_full (FP_DEVICE (dev), send_recv_ssm, SEND_RECV_STATES, SEND_RECV_STATES, "identify");
|
||||||
|
fpi_ssm_start (ssm, identify_cb);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
probe (FpDevice *dev)
|
||||||
|
{
|
||||||
|
g_auto(GStrv) argv = NULL;
|
||||||
|
FpDeviceVirtualSdcp *self = FPI_DEVICE_VIRTUAL_SDCP (dev);
|
||||||
|
GError *error = NULL;
|
||||||
|
const char *env;
|
||||||
|
|
||||||
|
/* We launch the test binary alread at probe time and quit only when
|
||||||
|
* the object is finalized. This allows testing reconnect properly.
|
||||||
|
*
|
||||||
|
* Also, we'll fail probe if something goes wrong executing it.
|
||||||
|
*/
|
||||||
|
env = fpi_device_get_virtual_env (FP_DEVICE (self));
|
||||||
|
|
||||||
|
if (!g_shell_parse_argv (env, NULL, &argv, &error))
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
self->proc = g_subprocess_newv ((const char * const *) argv,
|
||||||
|
G_SUBPROCESS_FLAGS_STDIN_PIPE | G_SUBPROCESS_FLAGS_STDOUT_PIPE,
|
||||||
|
&error);
|
||||||
|
if (!self->proc)
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
self->proc_stdin = g_object_ref (g_subprocess_get_stdin_pipe (self->proc));
|
||||||
|
self->proc_stdout = g_object_ref (g_subprocess_get_stdout_pipe (self->proc));
|
||||||
|
|
||||||
|
|
||||||
|
out:
|
||||||
|
fpi_device_probe_complete (dev, "virtual-sdcp", NULL, error);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
dev_close (FpDevice *dev)
|
||||||
|
{
|
||||||
|
/* No-op, needs to be defined. */
|
||||||
|
fpi_device_close_complete (dev, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
fpi_device_virtual_sdcp_init (FpDeviceVirtualSdcp *self)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
fpi_device_virtual_sdcp_finalize (GObject *obj)
|
||||||
|
{
|
||||||
|
FpDeviceVirtualSdcp *self = FPI_DEVICE_VIRTUAL_SDCP (obj);
|
||||||
|
|
||||||
|
/* Just kill the subprocess, no need to be graceful here. */
|
||||||
|
if (self->proc)
|
||||||
|
g_subprocess_force_exit (self->proc);
|
||||||
|
|
||||||
|
g_clear_object (&self->proc);
|
||||||
|
g_clear_object (&self->proc_stdin);
|
||||||
|
g_clear_object (&self->proc_stdout);
|
||||||
|
|
||||||
|
G_OBJECT_CLASS (fpi_device_virtual_sdcp_parent_class)->finalize (obj);
|
||||||
|
}
|
||||||
|
|
||||||
|
static const FpIdEntry driver_ids[] = {
|
||||||
|
{ .virtual_envvar = "FP_VIRTUAL_SDCP" },
|
||||||
|
{ .virtual_envvar = NULL }
|
||||||
|
};
|
||||||
|
|
||||||
|
static void
|
||||||
|
fpi_device_virtual_sdcp_class_init (FpDeviceVirtualSdcpClass *klass)
|
||||||
|
{
|
||||||
|
GObjectClass *obj_class = G_OBJECT_CLASS (klass);
|
||||||
|
FpDeviceClass *dev_class = FP_DEVICE_CLASS (klass);
|
||||||
|
FpSdcpDeviceClass *sdcp_class = FP_SDCP_DEVICE_CLASS (klass);
|
||||||
|
|
||||||
|
obj_class->finalize = fpi_device_virtual_sdcp_finalize;
|
||||||
|
|
||||||
|
dev_class->id = FP_COMPONENT;
|
||||||
|
dev_class->full_name = "Virtual SDCP device talking to MS test code";
|
||||||
|
dev_class->type = FP_DEVICE_TYPE_VIRTUAL;
|
||||||
|
dev_class->id_table = driver_ids;
|
||||||
|
dev_class->nr_enroll_stages = 1;
|
||||||
|
|
||||||
|
/* The SDCP base class may need to override this in the long run */
|
||||||
|
dev_class->probe = probe;
|
||||||
|
dev_class->close = dev_close;
|
||||||
|
|
||||||
|
sdcp_class->connect = connect;
|
||||||
|
sdcp_class->reconnect = reconnect;
|
||||||
|
|
||||||
|
sdcp_class->enroll_begin = enroll_begin;
|
||||||
|
sdcp_class->enroll_commit = enroll_commit;
|
||||||
|
|
||||||
|
sdcp_class->identify = identify;
|
||||||
|
}
|
|
@ -574,7 +574,7 @@ fp_context_enumerate (FpContext *context)
|
||||||
*
|
*
|
||||||
* Get all devices. fp_context_enumerate() will be called as needed.
|
* Get all devices. fp_context_enumerate() will be called as needed.
|
||||||
*
|
*
|
||||||
* Returns: (transfer none) (element-type FpDevice): a new #GPtrArray of #FpDevice's.
|
* Returns: (transfer none) (element-type FpDevice): a new #GPtrArray of #GUsbDevice's.
|
||||||
*/
|
*/
|
||||||
GPtrArray *
|
GPtrArray *
|
||||||
fp_context_get_devices (FpContext *context)
|
fp_context_get_devices (FpContext *context)
|
||||||
|
|
|
@ -22,23 +22,6 @@
|
||||||
|
|
||||||
#include "fpi-device.h"
|
#include "fpi-device.h"
|
||||||
|
|
||||||
/* Chosen so that if we turn on after WARM -> COLD, it takes exactly one time
|
|
||||||
* constant to go from COLD -> HOT.
|
|
||||||
* TEMP_COLD_THRESH = 1 / (e + 1)
|
|
||||||
*/
|
|
||||||
#define TEMP_COLD_THRESH (0.26894142136999512075)
|
|
||||||
#define TEMP_WARM_HOT_THRESH (1.0 - TEMP_COLD_THRESH)
|
|
||||||
#define TEMP_HOT_WARM_THRESH (0.5)
|
|
||||||
|
|
||||||
/* Delay updates by 100ms to avoid hitting the border exactly */
|
|
||||||
#define TEMP_DELAY_SECONDS 0.1
|
|
||||||
|
|
||||||
/* Hopefully 3min is long enough to not get in the way, while also not
|
|
||||||
* properly overheating any devices.
|
|
||||||
*/
|
|
||||||
#define DEFAULT_TEMP_HOT_SECONDS (3 * 60)
|
|
||||||
#define DEFAULT_TEMP_COLD_SECONDS (9 * 60)
|
|
||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
FpDeviceType type;
|
FpDeviceType type;
|
||||||
|
@ -53,7 +36,6 @@ typedef struct
|
||||||
|
|
||||||
gboolean is_removed;
|
gboolean is_removed;
|
||||||
gboolean is_open;
|
gboolean is_open;
|
||||||
gboolean is_suspended;
|
|
||||||
|
|
||||||
gchar *device_id;
|
gchar *device_id;
|
||||||
gchar *device_name;
|
gchar *device_name;
|
||||||
|
@ -68,37 +50,14 @@ typedef struct
|
||||||
/* We always make sure that only one task is run at a time. */
|
/* We always make sure that only one task is run at a time. */
|
||||||
FpiDeviceAction current_action;
|
FpiDeviceAction current_action;
|
||||||
GTask *current_task;
|
GTask *current_task;
|
||||||
GError *current_cancellation_reason;
|
|
||||||
GAsyncReadyCallback current_user_cb;
|
GAsyncReadyCallback current_user_cb;
|
||||||
GCancellable *current_cancellable;
|
|
||||||
gulong current_cancellable_id;
|
gulong current_cancellable_id;
|
||||||
gulong current_task_cancellable_id;
|
|
||||||
GSource *current_idle_cancel_source;
|
GSource *current_idle_cancel_source;
|
||||||
GSource *current_task_idle_return_source;
|
GSource *current_task_idle_return_source;
|
||||||
|
|
||||||
/* State for tasks */
|
/* State for tasks */
|
||||||
gboolean wait_for_finger;
|
gboolean wait_for_finger;
|
||||||
FpFingerStatusFlags finger_status;
|
FpFingerStatusFlags finger_status;
|
||||||
|
|
||||||
/* Driver critical sections */
|
|
||||||
guint critical_section;
|
|
||||||
GSource *critical_section_flush_source;
|
|
||||||
gboolean cancel_queued;
|
|
||||||
gboolean suspend_queued;
|
|
||||||
gboolean resume_queued;
|
|
||||||
|
|
||||||
/* Suspend/resume tasks */
|
|
||||||
GTask *suspend_resume_task;
|
|
||||||
GError *suspend_error;
|
|
||||||
|
|
||||||
/* Device temperature model information and state */
|
|
||||||
GSource *temp_timeout;
|
|
||||||
FpTemperature temp_current;
|
|
||||||
gint32 temp_hot_seconds;
|
|
||||||
gint32 temp_cold_seconds;
|
|
||||||
gint64 temp_last_update;
|
|
||||||
gboolean temp_last_active;
|
|
||||||
gdouble temp_current_ratio;
|
|
||||||
} FpDevicePrivate;
|
} FpDevicePrivate;
|
||||||
|
|
||||||
|
|
||||||
|
@ -129,8 +88,3 @@ typedef struct
|
||||||
} FpMatchData;
|
} FpMatchData;
|
||||||
|
|
||||||
void match_data_free (FpMatchData *match_data);
|
void match_data_free (FpMatchData *match_data);
|
||||||
|
|
||||||
void fpi_device_configure_wakeup (FpDevice *device,
|
|
||||||
gboolean enabled);
|
|
||||||
void fpi_device_update_temp (FpDevice *device,
|
|
||||||
gboolean is_active);
|
|
||||||
|
|
|
@ -48,7 +48,6 @@ enum {
|
||||||
PROP_NR_ENROLL_STAGES,
|
PROP_NR_ENROLL_STAGES,
|
||||||
PROP_SCAN_TYPE,
|
PROP_SCAN_TYPE,
|
||||||
PROP_FINGER_STATUS,
|
PROP_FINGER_STATUS,
|
||||||
PROP_TEMPERATURE,
|
|
||||||
PROP_FPI_ENVIRON,
|
PROP_FPI_ENVIRON,
|
||||||
PROP_FPI_USB_DEVICE,
|
PROP_FPI_USB_DEVICE,
|
||||||
PROP_FPI_UDEV_DATA_SPIDEV,
|
PROP_FPI_UDEV_DATA_SPIDEV,
|
||||||
|
@ -94,9 +93,6 @@ fp_device_cancel_in_idle_cb (gpointer user_data)
|
||||||
|
|
||||||
priv->current_idle_cancel_source = NULL;
|
priv->current_idle_cancel_source = NULL;
|
||||||
|
|
||||||
if (priv->critical_section)
|
|
||||||
priv->cancel_queued = TRUE;
|
|
||||||
else
|
|
||||||
cls->cancel (self);
|
cls->cancel (self);
|
||||||
|
|
||||||
fpi_device_report_finger_status (self, FP_FINGER_STATUS_NONE);
|
fpi_device_report_finger_status (self, FP_FINGER_STATUS_NONE);
|
||||||
|
@ -122,42 +118,22 @@ fp_device_cancelled_cb (GCancellable *cancellable, FpDevice *self)
|
||||||
g_source_unref (priv->current_idle_cancel_source);
|
g_source_unref (priv->current_idle_cancel_source);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Forward the external task cancellable to the internal one. */
|
|
||||||
static void
|
static void
|
||||||
fp_device_task_cancelled_cb (GCancellable *cancellable, FpDevice *self)
|
maybe_cancel_on_cancelled (FpDevice *device,
|
||||||
|
GCancellable *cancellable)
|
||||||
{
|
{
|
||||||
FpDevicePrivate *priv = fp_device_get_instance_private (self);
|
|
||||||
|
|
||||||
g_cancellable_cancel (priv->current_cancellable);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
setup_task_cancellable (FpDevice *device)
|
|
||||||
{
|
|
||||||
FpDevicePrivate *priv = fp_device_get_instance_private (device);
|
|
||||||
FpDeviceClass *cls = FP_DEVICE_GET_CLASS (device);
|
FpDeviceClass *cls = FP_DEVICE_GET_CLASS (device);
|
||||||
|
FpDevicePrivate *priv = fp_device_get_instance_private (device);
|
||||||
|
|
||||||
/* Create an internal cancellable and hook it up. */
|
if (!cancellable || !cls->cancel)
|
||||||
priv->current_cancellable = g_cancellable_new ();
|
return;
|
||||||
if (cls->cancel)
|
|
||||||
{
|
priv->current_cancellable_id = g_cancellable_connect (cancellable,
|
||||||
priv->current_cancellable_id = g_cancellable_connect (priv->current_cancellable,
|
|
||||||
G_CALLBACK (fp_device_cancelled_cb),
|
G_CALLBACK (fp_device_cancelled_cb),
|
||||||
device,
|
device,
|
||||||
NULL);
|
NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Task cancellable is the externally visible one, make our internal one
|
|
||||||
* a slave of the external one. */
|
|
||||||
if (g_task_get_cancellable (priv->current_task))
|
|
||||||
{
|
|
||||||
priv->current_task_cancellable_id = g_cancellable_connect (g_task_get_cancellable (priv->current_task),
|
|
||||||
G_CALLBACK (fp_device_task_cancelled_cb),
|
|
||||||
device,
|
|
||||||
NULL);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
fp_device_constructed (GObject *object)
|
fp_device_constructed (GObject *object)
|
||||||
{
|
{
|
||||||
|
@ -175,36 +151,6 @@ fp_device_constructed (GObject *object)
|
||||||
priv->device_name = g_strdup (cls->full_name);
|
priv->device_name = g_strdup (cls->full_name);
|
||||||
priv->device_id = g_strdup ("0");
|
priv->device_id = g_strdup ("0");
|
||||||
|
|
||||||
if (cls->temp_hot_seconds > 0)
|
|
||||||
{
|
|
||||||
priv->temp_hot_seconds = cls->temp_hot_seconds;
|
|
||||||
priv->temp_cold_seconds = cls->temp_cold_seconds;
|
|
||||||
g_assert (priv->temp_cold_seconds > 0);
|
|
||||||
}
|
|
||||||
else if (cls->temp_hot_seconds == 0)
|
|
||||||
{
|
|
||||||
priv->temp_hot_seconds = DEFAULT_TEMP_HOT_SECONDS;
|
|
||||||
priv->temp_cold_seconds = DEFAULT_TEMP_COLD_SECONDS;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
/* Temperature management disabled */
|
|
||||||
priv->temp_hot_seconds = -1;
|
|
||||||
priv->temp_cold_seconds = -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Start out at not completely cold (i.e. assume we are only at the upper
|
|
||||||
* bound of COLD).
|
|
||||||
* To be fair, the warm-up from 0 to WARM should be really short either way.
|
|
||||||
*
|
|
||||||
* Note that a call to fpi_device_update_temp() is not needed here as no
|
|
||||||
* timeout must be registered.
|
|
||||||
*/
|
|
||||||
priv->temp_current = FP_TEMPERATURE_COLD;
|
|
||||||
priv->temp_current_ratio = TEMP_COLD_THRESH;
|
|
||||||
priv->temp_last_update = g_get_monotonic_time ();
|
|
||||||
priv->temp_last_active = FALSE;
|
|
||||||
|
|
||||||
G_OBJECT_CLASS (fp_device_parent_class)->constructed (object);
|
G_OBJECT_CLASS (fp_device_parent_class)->constructed (object);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -219,8 +165,6 @@ fp_device_finalize (GObject *object)
|
||||||
if (priv->is_open)
|
if (priv->is_open)
|
||||||
g_warning ("User destroyed open device! Not cleaning up properly!");
|
g_warning ("User destroyed open device! Not cleaning up properly!");
|
||||||
|
|
||||||
g_clear_pointer (&priv->temp_timeout, g_source_destroy);
|
|
||||||
|
|
||||||
g_slist_free_full (priv->sources, (GDestroyNotify) g_source_destroy);
|
g_slist_free_full (priv->sources, (GDestroyNotify) g_source_destroy);
|
||||||
|
|
||||||
g_clear_pointer (&priv->current_idle_cancel_source, g_source_destroy);
|
g_clear_pointer (&priv->current_idle_cancel_source, g_source_destroy);
|
||||||
|
@ -245,7 +189,6 @@ fp_device_get_property (GObject *object,
|
||||||
{
|
{
|
||||||
FpDevice *self = FP_DEVICE (object);
|
FpDevice *self = FP_DEVICE (object);
|
||||||
FpDevicePrivate *priv = fp_device_get_instance_private (self);
|
FpDevicePrivate *priv = fp_device_get_instance_private (self);
|
||||||
FpDeviceClass *cls = FP_DEVICE_GET_CLASS (self);
|
|
||||||
|
|
||||||
switch (prop_id)
|
switch (prop_id)
|
||||||
{
|
{
|
||||||
|
@ -261,10 +204,6 @@ fp_device_get_property (GObject *object,
|
||||||
g_value_set_flags (value, priv->finger_status);
|
g_value_set_flags (value, priv->finger_status);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case PROP_TEMPERATURE:
|
|
||||||
g_value_set_enum (value, priv->temp_current);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PROP_DRIVER:
|
case PROP_DRIVER:
|
||||||
g_value_set_static_string (value, FP_DEVICE_GET_CLASS (self)->id);
|
g_value_set_static_string (value, FP_DEVICE_GET_CLASS (self)->id);
|
||||||
break;
|
break;
|
||||||
|
@ -285,24 +224,6 @@ fp_device_get_property (GObject *object,
|
||||||
g_value_set_boolean (value, priv->is_removed);
|
g_value_set_boolean (value, priv->is_removed);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case PROP_FPI_USB_DEVICE:
|
|
||||||
g_value_set_object (value, priv->usb_device);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PROP_FPI_UDEV_DATA_SPIDEV:
|
|
||||||
if (cls->type == FP_DEVICE_TYPE_UDEV)
|
|
||||||
g_value_set_string (value, g_strdup (priv->udev_data.spidev_path));
|
|
||||||
else
|
|
||||||
g_value_set_string (value, NULL);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case PROP_FPI_UDEV_DATA_HIDRAW:
|
|
||||||
if (cls->type == FP_DEVICE_TYPE_UDEV)
|
|
||||||
g_value_set_string (value, g_strdup (priv->udev_data.hidraw_path));
|
|
||||||
else
|
|
||||||
g_value_set_string (value, NULL);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
default:
|
||||||
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||||||
}
|
}
|
||||||
|
@ -358,24 +279,6 @@ fp_device_set_property (GObject *object,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
device_idle_probe_cb (FpDevice *self, gpointer user_data)
|
|
||||||
{
|
|
||||||
/* This should not be an idle handler, see comment where it is registered.
|
|
||||||
*
|
|
||||||
* This effectively disables USB "persist" for us, and possibly turns off
|
|
||||||
* USB wakeup if it was enabled for some reason.
|
|
||||||
*/
|
|
||||||
fpi_device_configure_wakeup (self, FALSE);
|
|
||||||
|
|
||||||
if (!FP_DEVICE_GET_CLASS (self)->probe)
|
|
||||||
fpi_device_probe_complete (self, NULL, NULL, NULL);
|
|
||||||
else
|
|
||||||
FP_DEVICE_GET_CLASS (self)->probe (self);
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
fp_device_async_initable_init_async (GAsyncInitable *initable,
|
fp_device_async_initable_init_async (GAsyncInitable *initable,
|
||||||
int io_priority,
|
int io_priority,
|
||||||
|
@ -395,16 +298,17 @@ fp_device_async_initable_init_async (GAsyncInitable *initable,
|
||||||
if (g_task_return_error_if_cancelled (task))
|
if (g_task_return_error_if_cancelled (task))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
if (!FP_DEVICE_GET_CLASS (self)->probe)
|
||||||
|
{
|
||||||
|
g_task_return_boolean (task, TRUE);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
priv->current_action = FPI_DEVICE_ACTION_PROBE;
|
priv->current_action = FPI_DEVICE_ACTION_PROBE;
|
||||||
priv->current_task = g_steal_pointer (&task);
|
priv->current_task = g_steal_pointer (&task);
|
||||||
setup_task_cancellable (self);
|
maybe_cancel_on_cancelled (self, cancellable);
|
||||||
|
|
||||||
/* We push this into an idle handler for compatibility with libgusb
|
FP_DEVICE_GET_CLASS (self)->probe (self);
|
||||||
* 0.3.7 and before.
|
|
||||||
* See https://github.com/hughsie/libgusb/pull/50
|
|
||||||
*/
|
|
||||||
g_source_set_name (fpi_device_add_timeout (self, 0, device_idle_probe_cb, NULL, NULL),
|
|
||||||
"libusb probe in idle");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
|
@ -454,13 +358,6 @@ fp_device_class_init (FpDeviceClass *klass)
|
||||||
FP_TYPE_FINGER_STATUS_FLAGS, FP_FINGER_STATUS_NONE,
|
FP_TYPE_FINGER_STATUS_FLAGS, FP_FINGER_STATUS_NONE,
|
||||||
G_PARAM_STATIC_STRINGS | G_PARAM_READABLE);
|
G_PARAM_STATIC_STRINGS | G_PARAM_READABLE);
|
||||||
|
|
||||||
properties[PROP_TEMPERATURE] =
|
|
||||||
g_param_spec_enum ("temperature",
|
|
||||||
"Temperature",
|
|
||||||
"The temperature estimation for device to prevent overheating.",
|
|
||||||
FP_TYPE_TEMPERATURE, FP_TEMPERATURE_COLD,
|
|
||||||
G_PARAM_STATIC_STRINGS | G_PARAM_READABLE);
|
|
||||||
|
|
||||||
properties[PROP_DRIVER] =
|
properties[PROP_DRIVER] =
|
||||||
g_param_spec_string ("driver",
|
g_param_spec_string ("driver",
|
||||||
"Driver",
|
"Driver",
|
||||||
|
@ -549,7 +446,7 @@ fp_device_class_init (FpDeviceClass *klass)
|
||||||
"USB Device",
|
"USB Device",
|
||||||
"Private: The USB device for the device",
|
"Private: The USB device for the device",
|
||||||
G_USB_TYPE_DEVICE,
|
G_USB_TYPE_DEVICE,
|
||||||
G_PARAM_STATIC_STRINGS | G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY);
|
G_PARAM_STATIC_STRINGS | G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY);
|
||||||
/**
|
/**
|
||||||
* FpDevice::fpi-udev-data-spidev: (skip)
|
* FpDevice::fpi-udev-data-spidev: (skip)
|
||||||
*
|
*
|
||||||
|
@ -562,7 +459,7 @@ fp_device_class_init (FpDeviceClass *klass)
|
||||||
"Udev data: spidev path",
|
"Udev data: spidev path",
|
||||||
"Private: The path to /dev/spidevN.M",
|
"Private: The path to /dev/spidevN.M",
|
||||||
NULL,
|
NULL,
|
||||||
G_PARAM_STATIC_STRINGS | G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY);
|
G_PARAM_STATIC_STRINGS | G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY);
|
||||||
/**
|
/**
|
||||||
* FpDevice::fpi-udev-data-hidraw: (skip)
|
* FpDevice::fpi-udev-data-hidraw: (skip)
|
||||||
*
|
*
|
||||||
|
@ -575,7 +472,7 @@ fp_device_class_init (FpDeviceClass *klass)
|
||||||
"Udev data: hidraw path",
|
"Udev data: hidraw path",
|
||||||
"Private: The path to /dev/hidrawN",
|
"Private: The path to /dev/hidrawN",
|
||||||
NULL,
|
NULL,
|
||||||
G_PARAM_STATIC_STRINGS | G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY);
|
G_PARAM_STATIC_STRINGS | G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* FpDevice::fpi-driver-data: (skip)
|
* FpDevice::fpi-driver-data: (skip)
|
||||||
|
@ -719,25 +616,6 @@ fp_device_get_nr_enroll_stages (FpDevice *device)
|
||||||
return priv->nr_enroll_stages;
|
return priv->nr_enroll_stages;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* fp_device_get_temperature:
|
|
||||||
* @device: A #FpDevice
|
|
||||||
*
|
|
||||||
* Retrieves simple temperature information for device. It is not possible
|
|
||||||
* to use a device when this is #FP_TEMPERATURE_HOT.
|
|
||||||
*
|
|
||||||
* Returns: The current temperature estimation.
|
|
||||||
*/
|
|
||||||
FpTemperature
|
|
||||||
fp_device_get_temperature (FpDevice *device)
|
|
||||||
{
|
|
||||||
FpDevicePrivate *priv = fp_device_get_instance_private (device);
|
|
||||||
|
|
||||||
g_return_val_if_fail (FP_IS_DEVICE (device), -1);
|
|
||||||
|
|
||||||
return priv->temp_current;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* fp_device_supports_identify:
|
* fp_device_supports_identify:
|
||||||
* @device: A #FpDevice
|
* @device: A #FpDevice
|
||||||
|
@ -830,7 +708,7 @@ fp_device_open (FpDevice *device,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (priv->current_task || priv->is_suspended)
|
if (priv->current_task)
|
||||||
{
|
{
|
||||||
g_task_return_error (task,
|
g_task_return_error (task,
|
||||||
fpi_device_error_new (FP_DEVICE_ERROR_BUSY));
|
fpi_device_error_new (FP_DEVICE_ERROR_BUSY));
|
||||||
|
@ -859,7 +737,7 @@ fp_device_open (FpDevice *device,
|
||||||
|
|
||||||
priv->current_action = FPI_DEVICE_ACTION_OPEN;
|
priv->current_action = FPI_DEVICE_ACTION_OPEN;
|
||||||
priv->current_task = g_steal_pointer (&task);
|
priv->current_task = g_steal_pointer (&task);
|
||||||
setup_task_cancellable (device);
|
maybe_cancel_on_cancelled (device, cancellable);
|
||||||
fpi_device_report_finger_status (device, FP_FINGER_STATUS_NONE);
|
fpi_device_report_finger_status (device, FP_FINGER_STATUS_NONE);
|
||||||
|
|
||||||
FP_DEVICE_GET_CLASS (device)->open (device);
|
FP_DEVICE_GET_CLASS (device)->open (device);
|
||||||
|
@ -915,7 +793,7 @@ fp_device_close (FpDevice *device,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (priv->current_task || priv->is_suspended)
|
if (priv->current_task)
|
||||||
{
|
{
|
||||||
g_task_return_error (task,
|
g_task_return_error (task,
|
||||||
fpi_device_error_new (FP_DEVICE_ERROR_BUSY));
|
fpi_device_error_new (FP_DEVICE_ERROR_BUSY));
|
||||||
|
@ -924,7 +802,7 @@ fp_device_close (FpDevice *device,
|
||||||
|
|
||||||
priv->current_action = FPI_DEVICE_ACTION_CLOSE;
|
priv->current_action = FPI_DEVICE_ACTION_CLOSE;
|
||||||
priv->current_task = g_steal_pointer (&task);
|
priv->current_task = g_steal_pointer (&task);
|
||||||
setup_task_cancellable (device);
|
maybe_cancel_on_cancelled (device, cancellable);
|
||||||
|
|
||||||
FP_DEVICE_GET_CLASS (device)->close (device);
|
FP_DEVICE_GET_CLASS (device)->close (device);
|
||||||
}
|
}
|
||||||
|
@ -948,230 +826,6 @@ fp_device_close_finish (FpDevice *device,
|
||||||
return g_task_propagate_boolean (G_TASK (result), error);
|
return g_task_propagate_boolean (G_TASK (result), error);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
complete_suspend_resume_task (FpDevice *device)
|
|
||||||
{
|
|
||||||
FpDevicePrivate *priv = fp_device_get_instance_private (device);
|
|
||||||
|
|
||||||
g_assert (priv->suspend_resume_task);
|
|
||||||
|
|
||||||
g_task_return_boolean (g_steal_pointer (&priv->suspend_resume_task), TRUE);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* fp_device_suspend:
|
|
||||||
* @device: a #FpDevice
|
|
||||||
* @cancellable: (nullable): a #GCancellable, or %NULL, currently not used
|
|
||||||
* @callback: the function to call on completion
|
|
||||||
* @user_data: the data to pass to @callback
|
|
||||||
*
|
|
||||||
* Prepare the device for system suspend. Retrieve the result with
|
|
||||||
* fp_device_suspend_finish().
|
|
||||||
*
|
|
||||||
* The suspend method can be called at any time (even if the device is not
|
|
||||||
* opened) and must be paired with a corresponding resume call. It is undefined
|
|
||||||
* when or how any ongoing operation is finished. This call might wait for an
|
|
||||||
* ongoing operation to finish, might cancel the ongoing operation or may
|
|
||||||
* prepare the device so that the host is resumed when the operation can be
|
|
||||||
* finished.
|
|
||||||
*
|
|
||||||
* If an ongoing operation must be cancelled then it will complete with an error
|
|
||||||
* code of #FP_DEVICE_ERROR_BUSY before the suspend async routine finishes.
|
|
||||||
*
|
|
||||||
* Any operation started while the device is suspended will fail with
|
|
||||||
* #FP_DEVICE_ERROR_BUSY, this includes calls to open or close the device.
|
|
||||||
*/
|
|
||||||
void
|
|
||||||
fp_device_suspend (FpDevice *device,
|
|
||||||
GCancellable *cancellable,
|
|
||||||
GAsyncReadyCallback callback,
|
|
||||||
gpointer user_data)
|
|
||||||
{
|
|
||||||
g_autoptr(GTask) task = NULL;
|
|
||||||
FpDevicePrivate *priv = fp_device_get_instance_private (device);
|
|
||||||
|
|
||||||
task = g_task_new (device, cancellable, callback, user_data);
|
|
||||||
|
|
||||||
if (priv->suspend_resume_task || priv->is_suspended)
|
|
||||||
{
|
|
||||||
g_task_return_error (task,
|
|
||||||
fpi_device_error_new (FP_DEVICE_ERROR_BUSY));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (priv->is_removed)
|
|
||||||
{
|
|
||||||
g_task_return_error (task,
|
|
||||||
fpi_device_error_new (FP_DEVICE_ERROR_REMOVED));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
priv->suspend_resume_task = g_steal_pointer (&task);
|
|
||||||
|
|
||||||
/* If the device is currently idle, just complete immediately.
|
|
||||||
* For long running tasks, call the driver handler right away, for short
|
|
||||||
* tasks, wait for completion and then return the task.
|
|
||||||
*/
|
|
||||||
switch (priv->current_action)
|
|
||||||
{
|
|
||||||
case FPI_DEVICE_ACTION_NONE:
|
|
||||||
fpi_device_suspend_complete (device, NULL);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case FPI_DEVICE_ACTION_ENROLL:
|
|
||||||
case FPI_DEVICE_ACTION_VERIFY:
|
|
||||||
case FPI_DEVICE_ACTION_IDENTIFY:
|
|
||||||
case FPI_DEVICE_ACTION_CAPTURE:
|
|
||||||
if (FP_DEVICE_GET_CLASS (device)->suspend)
|
|
||||||
{
|
|
||||||
if (priv->critical_section)
|
|
||||||
priv->suspend_queued = TRUE;
|
|
||||||
else
|
|
||||||
FP_DEVICE_GET_CLASS (device)->suspend (device);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
fpi_device_suspend_complete (device, fpi_device_error_new (FP_DEVICE_ERROR_NOT_SUPPORTED));
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
case FPI_DEVICE_ACTION_PROBE:
|
|
||||||
case FPI_DEVICE_ACTION_OPEN:
|
|
||||||
case FPI_DEVICE_ACTION_CLOSE:
|
|
||||||
case FPI_DEVICE_ACTION_DELETE:
|
|
||||||
case FPI_DEVICE_ACTION_LIST:
|
|
||||||
case FPI_DEVICE_ACTION_CLEAR_STORAGE:
|
|
||||||
g_signal_connect_object (priv->current_task,
|
|
||||||
"notify::completed",
|
|
||||||
G_CALLBACK (complete_suspend_resume_task),
|
|
||||||
device,
|
|
||||||
G_CONNECT_SWAPPED);
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* fp_device_suspend_finish:
|
|
||||||
* @device: A #FpDevice
|
|
||||||
* @result: A #GAsyncResult
|
|
||||||
* @error: Return location for errors, or %NULL to ignore
|
|
||||||
*
|
|
||||||
* Finish an asynchronous operation to prepare the device for suspend.
|
|
||||||
* See fp_device_suspend().
|
|
||||||
*
|
|
||||||
* The API user should accept an error of #FP_DEVICE_ERROR_NOT_SUPPORTED.
|
|
||||||
*
|
|
||||||
* Returns: (type void): %FALSE on error, %TRUE otherwise
|
|
||||||
*/
|
|
||||||
gboolean
|
|
||||||
fp_device_suspend_finish (FpDevice *device,
|
|
||||||
GAsyncResult *result,
|
|
||||||
GError **error)
|
|
||||||
{
|
|
||||||
return g_task_propagate_boolean (G_TASK (result), error);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* fp_device_resume:
|
|
||||||
* @device: a #FpDevice
|
|
||||||
* @cancellable: (nullable): a #GCancellable, or %NULL, currently not used
|
|
||||||
* @callback: the function to call on completion
|
|
||||||
* @user_data: the data to pass to @callback
|
|
||||||
*
|
|
||||||
* Resume device after system suspend. Retrieve the result with
|
|
||||||
* fp_device_suspend_finish().
|
|
||||||
*
|
|
||||||
* Note that it is not defined when any ongoing operation may return (success or
|
|
||||||
* error). You must be ready to handle this before, during or after the
|
|
||||||
* resume operation.
|
|
||||||
*/
|
|
||||||
void
|
|
||||||
fp_device_resume (FpDevice *device,
|
|
||||||
GCancellable *cancellable,
|
|
||||||
GAsyncReadyCallback callback,
|
|
||||||
gpointer user_data)
|
|
||||||
{
|
|
||||||
g_autoptr(GTask) task = NULL;
|
|
||||||
FpDevicePrivate *priv = fp_device_get_instance_private (device);
|
|
||||||
|
|
||||||
task = g_task_new (device, cancellable, callback, user_data);
|
|
||||||
|
|
||||||
if (priv->suspend_resume_task || !priv->is_suspended)
|
|
||||||
{
|
|
||||||
g_task_return_error (task,
|
|
||||||
fpi_device_error_new (FP_DEVICE_ERROR_BUSY));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (priv->is_removed)
|
|
||||||
{
|
|
||||||
g_task_return_error (task,
|
|
||||||
fpi_device_error_new (FP_DEVICE_ERROR_REMOVED));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
priv->suspend_resume_task = g_steal_pointer (&task);
|
|
||||||
|
|
||||||
switch (priv->current_action)
|
|
||||||
{
|
|
||||||
case FPI_DEVICE_ACTION_NONE:
|
|
||||||
fpi_device_resume_complete (device, NULL);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case FPI_DEVICE_ACTION_ENROLL:
|
|
||||||
case FPI_DEVICE_ACTION_VERIFY:
|
|
||||||
case FPI_DEVICE_ACTION_IDENTIFY:
|
|
||||||
case FPI_DEVICE_ACTION_CAPTURE:
|
|
||||||
if (FP_DEVICE_GET_CLASS (device)->resume)
|
|
||||||
{
|
|
||||||
if (priv->critical_section)
|
|
||||||
priv->resume_queued = TRUE;
|
|
||||||
else
|
|
||||||
FP_DEVICE_GET_CLASS (device)->resume (device);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
fpi_device_resume_complete (device, fpi_device_error_new (FP_DEVICE_ERROR_NOT_SUPPORTED));
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
case FPI_DEVICE_ACTION_PROBE:
|
|
||||||
case FPI_DEVICE_ACTION_OPEN:
|
|
||||||
case FPI_DEVICE_ACTION_CLOSE:
|
|
||||||
case FPI_DEVICE_ACTION_DELETE:
|
|
||||||
case FPI_DEVICE_ACTION_LIST:
|
|
||||||
case FPI_DEVICE_ACTION_CLEAR_STORAGE:
|
|
||||||
/* cannot happen as we make sure these tasks complete before suspend */
|
|
||||||
g_assert_not_reached ();
|
|
||||||
complete_suspend_resume_task (device);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* fp_device_resume_finish:
|
|
||||||
* @device: A #FpDevice
|
|
||||||
* @result: A #GAsyncResult
|
|
||||||
* @error: Return location for errors, or %NULL to ignore
|
|
||||||
*
|
|
||||||
* Finish an asynchronous operation to resume the device after suspend.
|
|
||||||
* See fp_device_resume().
|
|
||||||
*
|
|
||||||
* The API user should accept an error of #FP_DEVICE_ERROR_NOT_SUPPORTED.
|
|
||||||
*
|
|
||||||
* Returns: (type void): %FALSE on error, %TRUE otherwise
|
|
||||||
*/
|
|
||||||
gboolean
|
|
||||||
fp_device_resume_finish (FpDevice *device,
|
|
||||||
GAsyncResult *result,
|
|
||||||
GError **error)
|
|
||||||
{
|
|
||||||
return g_task_propagate_boolean (G_TASK (result), error);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* fp_device_enroll:
|
* fp_device_enroll:
|
||||||
|
@ -1220,7 +874,7 @@ fp_device_enroll (FpDevice *device,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (priv->current_task || priv->is_suspended)
|
if (priv->current_task)
|
||||||
{
|
{
|
||||||
g_task_return_error (task,
|
g_task_return_error (task,
|
||||||
fpi_device_error_new (FP_DEVICE_ERROR_BUSY));
|
fpi_device_error_new (FP_DEVICE_ERROR_BUSY));
|
||||||
|
@ -1246,15 +900,7 @@ fp_device_enroll (FpDevice *device,
|
||||||
|
|
||||||
priv->current_action = FPI_DEVICE_ACTION_ENROLL;
|
priv->current_action = FPI_DEVICE_ACTION_ENROLL;
|
||||||
priv->current_task = g_steal_pointer (&task);
|
priv->current_task = g_steal_pointer (&task);
|
||||||
setup_task_cancellable (device);
|
maybe_cancel_on_cancelled (device, cancellable);
|
||||||
|
|
||||||
fpi_device_update_temp (device, TRUE);
|
|
||||||
if (priv->temp_current == FP_TEMPERATURE_HOT)
|
|
||||||
{
|
|
||||||
g_task_return_error (task, fpi_device_error_new (FP_DEVICE_ERROR_TOO_HOT));
|
|
||||||
fpi_device_update_temp (device, FALSE);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
data = g_new0 (FpEnrollData, 1);
|
data = g_new0 (FpEnrollData, 1);
|
||||||
data->print = g_object_ref_sink (template_print);
|
data->print = g_object_ref_sink (template_print);
|
||||||
|
@ -1330,7 +976,7 @@ fp_device_verify (FpDevice *device,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (priv->current_task || priv->is_suspended)
|
if (priv->current_task)
|
||||||
{
|
{
|
||||||
g_task_return_error (task,
|
g_task_return_error (task,
|
||||||
fpi_device_error_new (FP_DEVICE_ERROR_BUSY));
|
fpi_device_error_new (FP_DEVICE_ERROR_BUSY));
|
||||||
|
@ -1347,15 +993,7 @@ fp_device_verify (FpDevice *device,
|
||||||
|
|
||||||
priv->current_action = FPI_DEVICE_ACTION_VERIFY;
|
priv->current_action = FPI_DEVICE_ACTION_VERIFY;
|
||||||
priv->current_task = g_steal_pointer (&task);
|
priv->current_task = g_steal_pointer (&task);
|
||||||
setup_task_cancellable (device);
|
maybe_cancel_on_cancelled (device, cancellable);
|
||||||
|
|
||||||
fpi_device_update_temp (device, TRUE);
|
|
||||||
if (priv->temp_current == FP_TEMPERATURE_HOT)
|
|
||||||
{
|
|
||||||
g_task_return_error (task, fpi_device_error_new (FP_DEVICE_ERROR_TOO_HOT));
|
|
||||||
fpi_device_update_temp (device, FALSE);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
data = g_new0 (FpMatchData, 1);
|
data = g_new0 (FpMatchData, 1);
|
||||||
data->enrolled_print = g_object_ref (enrolled_print);
|
data->enrolled_print = g_object_ref (enrolled_print);
|
||||||
|
@ -1457,7 +1095,7 @@ fp_device_identify (FpDevice *device,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (priv->current_task || priv->is_suspended)
|
if (priv->current_task)
|
||||||
{
|
{
|
||||||
g_task_return_error (task,
|
g_task_return_error (task,
|
||||||
fpi_device_error_new (FP_DEVICE_ERROR_BUSY));
|
fpi_device_error_new (FP_DEVICE_ERROR_BUSY));
|
||||||
|
@ -1474,15 +1112,7 @@ fp_device_identify (FpDevice *device,
|
||||||
|
|
||||||
priv->current_action = FPI_DEVICE_ACTION_IDENTIFY;
|
priv->current_action = FPI_DEVICE_ACTION_IDENTIFY;
|
||||||
priv->current_task = g_steal_pointer (&task);
|
priv->current_task = g_steal_pointer (&task);
|
||||||
setup_task_cancellable (device);
|
maybe_cancel_on_cancelled (device, cancellable);
|
||||||
|
|
||||||
fpi_device_update_temp (device, TRUE);
|
|
||||||
if (priv->temp_current == FP_TEMPERATURE_HOT)
|
|
||||||
{
|
|
||||||
g_task_return_error (task, fpi_device_error_new (FP_DEVICE_ERROR_TOO_HOT));
|
|
||||||
fpi_device_update_temp (device, FALSE);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
data = g_new0 (FpMatchData, 1);
|
data = g_new0 (FpMatchData, 1);
|
||||||
/* We cannot store the gallery directly, because the ptr array may not own
|
/* We cannot store the gallery directly, because the ptr array may not own
|
||||||
|
@ -1582,7 +1212,7 @@ fp_device_capture (FpDevice *device,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (priv->current_task || priv->is_suspended)
|
if (priv->current_task)
|
||||||
{
|
{
|
||||||
g_task_return_error (task,
|
g_task_return_error (task,
|
||||||
fpi_device_error_new (FP_DEVICE_ERROR_BUSY));
|
fpi_device_error_new (FP_DEVICE_ERROR_BUSY));
|
||||||
|
@ -1599,15 +1229,7 @@ fp_device_capture (FpDevice *device,
|
||||||
|
|
||||||
priv->current_action = FPI_DEVICE_ACTION_CAPTURE;
|
priv->current_action = FPI_DEVICE_ACTION_CAPTURE;
|
||||||
priv->current_task = g_steal_pointer (&task);
|
priv->current_task = g_steal_pointer (&task);
|
||||||
setup_task_cancellable (device);
|
maybe_cancel_on_cancelled (device, cancellable);
|
||||||
|
|
||||||
fpi_device_update_temp (device, TRUE);
|
|
||||||
if (priv->temp_current == FP_TEMPERATURE_HOT)
|
|
||||||
{
|
|
||||||
g_task_return_error (task, fpi_device_error_new (FP_DEVICE_ERROR_TOO_HOT));
|
|
||||||
fpi_device_update_temp (device, FALSE);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
priv->wait_for_finger = wait_for_finger;
|
priv->wait_for_finger = wait_for_finger;
|
||||||
|
|
||||||
|
@ -1673,7 +1295,7 @@ fp_device_delete_print (FpDevice *device,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (priv->current_task || priv->is_suspended)
|
if (priv->current_task)
|
||||||
{
|
{
|
||||||
g_task_return_error (task,
|
g_task_return_error (task,
|
||||||
fpi_device_error_new (FP_DEVICE_ERROR_BUSY));
|
fpi_device_error_new (FP_DEVICE_ERROR_BUSY));
|
||||||
|
@ -1689,7 +1311,7 @@ fp_device_delete_print (FpDevice *device,
|
||||||
|
|
||||||
priv->current_action = FPI_DEVICE_ACTION_DELETE;
|
priv->current_action = FPI_DEVICE_ACTION_DELETE;
|
||||||
priv->current_task = g_steal_pointer (&task);
|
priv->current_task = g_steal_pointer (&task);
|
||||||
setup_task_cancellable (device);
|
maybe_cancel_on_cancelled (device, cancellable);
|
||||||
|
|
||||||
g_task_set_task_data (priv->current_task,
|
g_task_set_task_data (priv->current_task,
|
||||||
g_object_ref (enrolled_print),
|
g_object_ref (enrolled_print),
|
||||||
|
@ -1751,7 +1373,7 @@ fp_device_list_prints (FpDevice *device,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (priv->current_task || priv->is_suspended)
|
if (priv->current_task)
|
||||||
{
|
{
|
||||||
g_task_return_error (task,
|
g_task_return_error (task,
|
||||||
fpi_device_error_new (FP_DEVICE_ERROR_BUSY));
|
fpi_device_error_new (FP_DEVICE_ERROR_BUSY));
|
||||||
|
@ -1768,7 +1390,7 @@ fp_device_list_prints (FpDevice *device,
|
||||||
|
|
||||||
priv->current_action = FPI_DEVICE_ACTION_LIST;
|
priv->current_action = FPI_DEVICE_ACTION_LIST;
|
||||||
priv->current_task = g_steal_pointer (&task);
|
priv->current_task = g_steal_pointer (&task);
|
||||||
setup_task_cancellable (device);
|
maybe_cancel_on_cancelled (device, cancellable);
|
||||||
|
|
||||||
cls->list (device);
|
cls->list (device);
|
||||||
}
|
}
|
||||||
|
@ -1853,7 +1475,7 @@ fp_device_clear_storage (FpDevice *device,
|
||||||
|
|
||||||
priv->current_action = FPI_DEVICE_ACTION_CLEAR_STORAGE;
|
priv->current_action = FPI_DEVICE_ACTION_CLEAR_STORAGE;
|
||||||
priv->current_task = g_steal_pointer (&task);
|
priv->current_task = g_steal_pointer (&task);
|
||||||
setup_task_cancellable (device);
|
maybe_cancel_on_cancelled (device, cancellable);
|
||||||
|
|
||||||
cls->clear_storage (device);
|
cls->clear_storage (device);
|
||||||
|
|
||||||
|
@ -2147,7 +1769,6 @@ fp_device_list_prints_sync (FpDevice *device,
|
||||||
return fp_device_list_prints_finish (device, task, error);
|
return fp_device_list_prints_finish (device, task, error);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* fp_device_clear_storage_sync:
|
* fp_device_clear_storage_sync:
|
||||||
* @device: a #FpDevice
|
* @device: a #FpDevice
|
||||||
|
@ -2176,58 +1797,6 @@ fp_device_clear_storage_sync (FpDevice *device,
|
||||||
return fp_device_clear_storage_finish (device, task, error);
|
return fp_device_clear_storage_finish (device, task, error);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* fp_device_suspend_sync:
|
|
||||||
* @device: a #FpDevice
|
|
||||||
* @cancellable: (nullable): a #GCancellable, or %NULL, currently not used
|
|
||||||
* @error: Return location for errors, or %NULL to ignore
|
|
||||||
*
|
|
||||||
* Prepare device for suspend.
|
|
||||||
*
|
|
||||||
* Returns: (type void): %FALSE on error, %TRUE otherwise
|
|
||||||
*/
|
|
||||||
gboolean
|
|
||||||
fp_device_suspend_sync (FpDevice *device,
|
|
||||||
GCancellable *cancellable,
|
|
||||||
GError **error)
|
|
||||||
{
|
|
||||||
g_autoptr(GAsyncResult) task = NULL;
|
|
||||||
|
|
||||||
g_return_val_if_fail (FP_IS_DEVICE (device), FALSE);
|
|
||||||
|
|
||||||
fp_device_suspend (device, cancellable, async_result_ready, &task);
|
|
||||||
while (!task)
|
|
||||||
g_main_context_iteration (NULL, TRUE);
|
|
||||||
|
|
||||||
return fp_device_suspend_finish (device, task, error);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* fp_device_resume_sync:
|
|
||||||
* @device: a #FpDevice
|
|
||||||
* @cancellable: (nullable): a #GCancellable, or %NULL, currently not used
|
|
||||||
* @error: Return location for errors, or %NULL to ignore
|
|
||||||
*
|
|
||||||
* Resume device after suspend.
|
|
||||||
*
|
|
||||||
* Returns: (type void): %FALSE on error, %TRUE otherwise
|
|
||||||
*/
|
|
||||||
gboolean
|
|
||||||
fp_device_resume_sync (FpDevice *device,
|
|
||||||
GCancellable *cancellable,
|
|
||||||
GError **error)
|
|
||||||
{
|
|
||||||
g_autoptr(GAsyncResult) task = NULL;
|
|
||||||
|
|
||||||
g_return_val_if_fail (FP_IS_DEVICE (device), FALSE);
|
|
||||||
|
|
||||||
fp_device_resume (device, cancellable, async_result_ready, &task);
|
|
||||||
while (!task)
|
|
||||||
g_main_context_iteration (NULL, TRUE);
|
|
||||||
|
|
||||||
return fp_device_resume_finish (device, task, error);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* fp_device_get_features:
|
* fp_device_get_features:
|
||||||
* @device: a #FpDevice
|
* @device: a #FpDevice
|
||||||
|
|
|
@ -58,7 +58,6 @@ typedef enum {
|
||||||
* @FP_DEVICE_FEATURE_STORAGE_DELETE: Supports deleting stored templates
|
* @FP_DEVICE_FEATURE_STORAGE_DELETE: Supports deleting stored templates
|
||||||
* @FP_DEVICE_FEATURE_STORAGE_CLEAR: Supports clearing the whole storage
|
* @FP_DEVICE_FEATURE_STORAGE_CLEAR: Supports clearing the whole storage
|
||||||
* @FP_DEVICE_FEATURE_DUPLICATES_CHECK: Natively supports duplicates detection
|
* @FP_DEVICE_FEATURE_DUPLICATES_CHECK: Natively supports duplicates detection
|
||||||
* @FP_DEVICE_FEATURE_ALWAYS_ON: Whether the device can run continuously
|
|
||||||
*/
|
*/
|
||||||
typedef enum /*< flags >*/ {
|
typedef enum /*< flags >*/ {
|
||||||
FP_DEVICE_FEATURE_NONE = 0,
|
FP_DEVICE_FEATURE_NONE = 0,
|
||||||
|
@ -70,7 +69,6 @@ typedef enum /*< flags >*/ {
|
||||||
FP_DEVICE_FEATURE_STORAGE_DELETE = 1 << 5,
|
FP_DEVICE_FEATURE_STORAGE_DELETE = 1 << 5,
|
||||||
FP_DEVICE_FEATURE_STORAGE_CLEAR = 1 << 6,
|
FP_DEVICE_FEATURE_STORAGE_CLEAR = 1 << 6,
|
||||||
FP_DEVICE_FEATURE_DUPLICATES_CHECK = 1 << 7,
|
FP_DEVICE_FEATURE_DUPLICATES_CHECK = 1 << 7,
|
||||||
FP_DEVICE_FEATURE_ALWAYS_ON = 1 << 8,
|
|
||||||
} FpDeviceFeature;
|
} FpDeviceFeature;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -83,23 +81,6 @@ typedef enum {
|
||||||
FP_SCAN_TYPE_PRESS,
|
FP_SCAN_TYPE_PRESS,
|
||||||
} FpScanType;
|
} FpScanType;
|
||||||
|
|
||||||
/**
|
|
||||||
* FpTemperature:
|
|
||||||
* @FP_TEMPERATURE_COLD: Sensor is considered cold.
|
|
||||||
* @FP_TEMPERATURE_WARM: Sensor is warm, usage time may be limited.
|
|
||||||
* @FP_TEMPERATURE_HOT: Sensor is hot and cannot be used.
|
|
||||||
*
|
|
||||||
* When a device is created, it is assumed to be cold. Applications such as
|
|
||||||
* fprintd may want to ensure all devices on the system are cold before
|
|
||||||
* shutting down in order to ensure that the cool-off period is not violated
|
|
||||||
* because the internal libfprint state about the device is lost.
|
|
||||||
*/
|
|
||||||
typedef enum {
|
|
||||||
FP_TEMPERATURE_COLD,
|
|
||||||
FP_TEMPERATURE_WARM,
|
|
||||||
FP_TEMPERATURE_HOT,
|
|
||||||
} FpTemperature;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* FpDeviceRetry:
|
* FpDeviceRetry:
|
||||||
* @FP_DEVICE_RETRY_GENERAL: The scan did not succeed due to poor scan quality
|
* @FP_DEVICE_RETRY_GENERAL: The scan did not succeed due to poor scan quality
|
||||||
|
@ -137,7 +118,7 @@ typedef enum {
|
||||||
* @FP_DEVICE_ERROR_DATA_FULL: No space on device available for operation
|
* @FP_DEVICE_ERROR_DATA_FULL: No space on device available for operation
|
||||||
* @FP_DEVICE_ERROR_DATA_DUPLICATE: Enrolling template duplicates storaged templates
|
* @FP_DEVICE_ERROR_DATA_DUPLICATE: Enrolling template duplicates storaged templates
|
||||||
* @FP_DEVICE_ERROR_REMOVED: The device has been removed.
|
* @FP_DEVICE_ERROR_REMOVED: The device has been removed.
|
||||||
* @FP_DEVICE_ERROR_TOO_HOT: The device might be getting too hot
|
* @FP_DEVICE_ERROR_UNTRUSTED: Device cannot be trusted
|
||||||
*
|
*
|
||||||
* Error codes for device operations. More specific errors from other domains
|
* Error codes for device operations. More specific errors from other domains
|
||||||
* such as #G_IO_ERROR or #G_USB_DEVICE_ERROR may also be reported.
|
* such as #G_IO_ERROR or #G_USB_DEVICE_ERROR may also be reported.
|
||||||
|
@ -155,7 +136,7 @@ typedef enum {
|
||||||
FP_DEVICE_ERROR_DATA_DUPLICATE,
|
FP_DEVICE_ERROR_DATA_DUPLICATE,
|
||||||
/* Leave some room to add more DATA related errors */
|
/* Leave some room to add more DATA related errors */
|
||||||
FP_DEVICE_ERROR_REMOVED = 0x100,
|
FP_DEVICE_ERROR_REMOVED = 0x100,
|
||||||
FP_DEVICE_ERROR_TOO_HOT,
|
FP_DEVICE_ERROR_UNTRUSTED,
|
||||||
} FpDeviceError;
|
} FpDeviceError;
|
||||||
|
|
||||||
GQuark fp_device_retry_quark (void);
|
GQuark fp_device_retry_quark (void);
|
||||||
|
@ -222,7 +203,6 @@ gboolean fp_device_is_open (FpDevice *device);
|
||||||
FpScanType fp_device_get_scan_type (FpDevice *device);
|
FpScanType fp_device_get_scan_type (FpDevice *device);
|
||||||
FpFingerStatusFlags fp_device_get_finger_status (FpDevice *device);
|
FpFingerStatusFlags fp_device_get_finger_status (FpDevice *device);
|
||||||
gint fp_device_get_nr_enroll_stages (FpDevice *device);
|
gint fp_device_get_nr_enroll_stages (FpDevice *device);
|
||||||
FpTemperature fp_device_get_temperature (FpDevice *device);
|
|
||||||
|
|
||||||
FpDeviceFeature fp_device_get_features (FpDevice *device);
|
FpDeviceFeature fp_device_get_features (FpDevice *device);
|
||||||
gboolean fp_device_has_feature (FpDevice *device,
|
gboolean fp_device_has_feature (FpDevice *device,
|
||||||
|
@ -239,16 +219,6 @@ void fp_device_close (FpDevice *device,
|
||||||
GAsyncReadyCallback callback,
|
GAsyncReadyCallback callback,
|
||||||
gpointer user_data);
|
gpointer user_data);
|
||||||
|
|
||||||
void fp_device_suspend (FpDevice *device,
|
|
||||||
GCancellable *cancellable,
|
|
||||||
GAsyncReadyCallback callback,
|
|
||||||
gpointer user_data);
|
|
||||||
|
|
||||||
void fp_device_resume (FpDevice *device,
|
|
||||||
GCancellable *cancellable,
|
|
||||||
GAsyncReadyCallback callback,
|
|
||||||
gpointer user_data);
|
|
||||||
|
|
||||||
void fp_device_enroll (FpDevice *device,
|
void fp_device_enroll (FpDevice *device,
|
||||||
FpPrint *template_print,
|
FpPrint *template_print,
|
||||||
GCancellable *cancellable,
|
GCancellable *cancellable,
|
||||||
|
@ -304,12 +274,6 @@ gboolean fp_device_open_finish (FpDevice *device,
|
||||||
gboolean fp_device_close_finish (FpDevice *device,
|
gboolean fp_device_close_finish (FpDevice *device,
|
||||||
GAsyncResult *result,
|
GAsyncResult *result,
|
||||||
GError **error);
|
GError **error);
|
||||||
gboolean fp_device_suspend_finish (FpDevice *device,
|
|
||||||
GAsyncResult *result,
|
|
||||||
GError **error);
|
|
||||||
gboolean fp_device_resume_finish (FpDevice *device,
|
|
||||||
GAsyncResult *result,
|
|
||||||
GError **error);
|
|
||||||
FpPrint *fp_device_enroll_finish (FpDevice *device,
|
FpPrint *fp_device_enroll_finish (FpDevice *device,
|
||||||
GAsyncResult *result,
|
GAsyncResult *result,
|
||||||
GError **error);
|
GError **error);
|
||||||
|
@ -378,13 +342,6 @@ GPtrArray * fp_device_list_prints_sync (FpDevice *device,
|
||||||
gboolean fp_device_clear_storage_sync (FpDevice *device,
|
gboolean fp_device_clear_storage_sync (FpDevice *device,
|
||||||
GCancellable *cancellable,
|
GCancellable *cancellable,
|
||||||
GError **error);
|
GError **error);
|
||||||
gboolean fp_device_suspend_sync (FpDevice *device,
|
|
||||||
GCancellable *cancellable,
|
|
||||||
GError **error);
|
|
||||||
gboolean fp_device_resume_sync (FpDevice *device,
|
|
||||||
GCancellable *cancellable,
|
|
||||||
GError **error);
|
|
||||||
|
|
||||||
/* Deprecated functions */
|
/* Deprecated functions */
|
||||||
G_DEPRECATED_FOR (fp_device_get_features)
|
G_DEPRECATED_FOR (fp_device_get_features)
|
||||||
gboolean fp_device_supports_identify (FpDevice *device);
|
gboolean fp_device_supports_identify (FpDevice *device);
|
||||||
|
|
|
@ -582,7 +582,7 @@ fp_print_equal (FpPrint *self, FpPrint *other)
|
||||||
if (g_strcmp0 (self->device_id, other->device_id))
|
if (g_strcmp0 (self->device_id, other->device_id))
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
if (self->type == FPI_PRINT_RAW)
|
if (self->type == FPI_PRINT_RAW || self->type == FPI_PRINT_SDCP)
|
||||||
{
|
{
|
||||||
return g_variant_equal (self->data, other->data);
|
return g_variant_equal (self->data, other->data);
|
||||||
}
|
}
|
||||||
|
@ -847,7 +847,7 @@ fp_print_deserialize (const guchar *data,
|
||||||
g_ptr_array_add (result->prints, g_steal_pointer (&xyt));
|
g_ptr_array_add (result->prints, g_steal_pointer (&xyt));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (type == FPI_PRINT_RAW)
|
else if (type == FPI_PRINT_RAW || type == FPI_PRINT_SDCP)
|
||||||
{
|
{
|
||||||
g_autoptr(GVariant) fp_data = g_variant_get_child_value (print_data, 0);
|
g_autoptr(GVariant) fp_data = g_variant_get_child_value (print_data, 0);
|
||||||
|
|
||||||
|
|
58
libfprint/fp-sdcp-device-private.h
Normal file
58
libfprint/fp-sdcp-device-private.h
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
/*
|
||||||
|
* FpSdcpDevice - A base class for SDCP enabled devices
|
||||||
|
* Copyright (C) 2020 Benjamin Berg <bberg@redhat.com>
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2.1 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "fpi-sdcp-device.h"
|
||||||
|
|
||||||
|
#include <nss.h>
|
||||||
|
#include <keyhi.h>
|
||||||
|
#include <keythi.h>
|
||||||
|
#include <pk11pub.h>
|
||||||
|
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
GError *enroll_pre_commit_error;
|
||||||
|
|
||||||
|
/* XXX: Do we want a separate SDCP session object?
|
||||||
|
*/
|
||||||
|
|
||||||
|
GPtrArray *intermediate_cas;
|
||||||
|
|
||||||
|
/* Host random for the connection */
|
||||||
|
guint8 host_random[32];
|
||||||
|
|
||||||
|
NSSInitContext *nss_init_context;
|
||||||
|
PK11SlotInfo *slot;
|
||||||
|
SECKEYPrivateKey *host_key_private;
|
||||||
|
SECKEYPublicKey *host_key_public;
|
||||||
|
|
||||||
|
/* Master secret is required for reconnects.
|
||||||
|
* TODO: We probably want to serialize this to disk so it can survive
|
||||||
|
* fprintd idle shutdown. */
|
||||||
|
PK11SymKey *master_secret;
|
||||||
|
PK11SymKey *mac_secret;
|
||||||
|
|
||||||
|
} FpSdcpDevicePrivate;
|
||||||
|
|
||||||
|
void fpi_sdcp_device_connect (FpSdcpDevice *self);
|
||||||
|
void fpi_sdcp_device_reconnect (FpSdcpDevice *self);
|
||||||
|
|
||||||
|
void fpi_sdcp_device_enroll (FpSdcpDevice *self);
|
||||||
|
void fpi_sdcp_device_identify (FpSdcpDevice *self);
|
141
libfprint/fp-sdcp-device.c
Normal file
141
libfprint/fp-sdcp-device.c
Normal file
|
@ -0,0 +1,141 @@
|
||||||
|
/*
|
||||||
|
* FpSdcpDevice - A base class for SDCP enabled devices
|
||||||
|
* Copyright (C) 2020 Benjamin Berg <bberg@redhat.com>
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2.1 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define FP_COMPONENT "sdcp_device"
|
||||||
|
#include "fpi-log.h"
|
||||||
|
|
||||||
|
#include "fp-sdcp-device-private.h"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* SECTION: fp-sdcp-device
|
||||||
|
* @title: FpSdcpDevice
|
||||||
|
* @short_description: SDCP device subclass
|
||||||
|
*
|
||||||
|
* This is a base class for devices implementing the SDCP security protocol.
|
||||||
|
*/
|
||||||
|
|
||||||
|
G_DEFINE_ABSTRACT_TYPE_WITH_PRIVATE (FpSdcpDevice, fp_sdcp_device, FP_TYPE_DEVICE)
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
/* XXX: We'll very likely want/need some properties on this class. */
|
||||||
|
enum {
|
||||||
|
PROP_0,
|
||||||
|
N_PROPS
|
||||||
|
};
|
||||||
|
|
||||||
|
static GParamSpec *properties[N_PROPS];
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*******************************************************/
|
||||||
|
|
||||||
|
/* Callbacks/vfuncs */
|
||||||
|
static void
|
||||||
|
fp_sdcp_device_open (FpDevice *device)
|
||||||
|
{
|
||||||
|
FpSdcpDevice *self = FP_SDCP_DEVICE (device);
|
||||||
|
FpSdcpDevicePrivate *priv = fp_sdcp_device_get_instance_private (self);
|
||||||
|
|
||||||
|
/* Try a reconnect if we still have the mac secret. */
|
||||||
|
if (priv->mac_secret)
|
||||||
|
fpi_sdcp_device_reconnect (self);
|
||||||
|
else
|
||||||
|
fpi_sdcp_device_connect (self);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
fp_sdcp_device_enroll (FpDevice *device)
|
||||||
|
{
|
||||||
|
FpSdcpDevice *self = FP_SDCP_DEVICE (device);
|
||||||
|
|
||||||
|
fpi_sdcp_device_enroll (self);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
fp_sdcp_device_identify (FpDevice *device)
|
||||||
|
{
|
||||||
|
FpSdcpDevice *self = FP_SDCP_DEVICE (device);
|
||||||
|
|
||||||
|
fpi_sdcp_device_identify (self);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*********************************************************/
|
||||||
|
|
||||||
|
static void
|
||||||
|
fp_sdcp_device_finalize (GObject *object)
|
||||||
|
{
|
||||||
|
FpSdcpDevice *self = (FpSdcpDevice *) object;
|
||||||
|
FpSdcpDevicePrivate *priv = fp_sdcp_device_get_instance_private (self);
|
||||||
|
|
||||||
|
g_clear_pointer (&priv->intermediate_cas, g_ptr_array_unref);
|
||||||
|
g_clear_pointer (&priv->slot, PK11_FreeSlot);
|
||||||
|
g_clear_pointer (&priv->host_key_private, SECKEY_DestroyPrivateKey);
|
||||||
|
g_clear_pointer (&priv->host_key_public, SECKEY_DestroyPublicKey);
|
||||||
|
g_clear_pointer (&priv->master_secret, PK11_FreeSymKey);
|
||||||
|
g_clear_pointer (&priv->mac_secret, PK11_FreeSymKey);
|
||||||
|
g_clear_pointer (&priv->nss_init_context, NSS_ShutdownContext);
|
||||||
|
|
||||||
|
G_OBJECT_CLASS (fp_sdcp_device_parent_class)->finalize (object);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
fp_sdcp_device_get_property (GObject *object,
|
||||||
|
guint prop_id,
|
||||||
|
GValue *value,
|
||||||
|
GParamSpec *pspec)
|
||||||
|
{
|
||||||
|
switch (prop_id)
|
||||||
|
{
|
||||||
|
default:
|
||||||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
fp_sdcp_device_constructed (GObject *obj)
|
||||||
|
{
|
||||||
|
G_OBJECT_CLASS (fp_sdcp_device_parent_class)->constructed (obj);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
fp_sdcp_device_class_init (FpSdcpDeviceClass *klass)
|
||||||
|
{
|
||||||
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
||||||
|
FpDeviceClass *fp_device_class = FP_DEVICE_CLASS (klass);
|
||||||
|
|
||||||
|
object_class->finalize = fp_sdcp_device_finalize;
|
||||||
|
object_class->get_property = fp_sdcp_device_get_property;
|
||||||
|
object_class->constructed = fp_sdcp_device_constructed;
|
||||||
|
|
||||||
|
fp_device_class->open = fp_sdcp_device_open;
|
||||||
|
fp_device_class->enroll = fp_sdcp_device_enroll;
|
||||||
|
fp_device_class->verify = fp_sdcp_device_identify;
|
||||||
|
fp_device_class->identify = fp_sdcp_device_identify;
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
g_object_class_install_properties (object_class, N_PROPS, properties);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
fp_sdcp_device_init (FpSdcpDevice *self)
|
||||||
|
{
|
||||||
|
FpSdcpDevicePrivate *priv = fp_sdcp_device_get_instance_private (self);
|
||||||
|
|
||||||
|
priv->intermediate_cas = g_ptr_array_new_with_free_func ((GDestroyNotify) g_bytes_unref);
|
||||||
|
}
|
29
libfprint/fp-sdcp-device.h
Normal file
29
libfprint/fp-sdcp-device.h
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
/*
|
||||||
|
* FpSdcpDevice - A base class for SDCP enabled devices
|
||||||
|
* Copyright (C) 2020 Benjamin Berg <bberg@redhat.com>
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2.1 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <fp-device.h>
|
||||||
|
|
||||||
|
G_BEGIN_DECLS
|
||||||
|
|
||||||
|
#define FP_TYPE_SDCP_DEVICE (fp_sdcp_device_get_type ())
|
||||||
|
G_DECLARE_DERIVABLE_TYPE (FpSdcpDevice, fp_sdcp_device, FP, SDCP_DEVICE, FpDevice)
|
||||||
|
|
||||||
|
G_END_DECLS
|
|
@ -19,9 +19,6 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#define FP_COMPONENT "device"
|
#define FP_COMPONENT "device"
|
||||||
#include <math.h>
|
|
||||||
#include <fcntl.h>
|
|
||||||
|
|
||||||
#include "fpi-log.h"
|
#include "fpi-log.h"
|
||||||
|
|
||||||
#include "fp-device-private.h"
|
#include "fp-device-private.h"
|
||||||
|
@ -81,9 +78,6 @@ fpi_device_class_auto_initialize_features (FpDeviceClass *device_class)
|
||||||
|
|
||||||
if (device_class->delete && (device_class->list || device_class->clear_storage))
|
if (device_class->delete && (device_class->list || device_class->clear_storage))
|
||||||
device_class->features |= FP_DEVICE_FEATURE_STORAGE;
|
device_class->features |= FP_DEVICE_FEATURE_STORAGE;
|
||||||
|
|
||||||
if (device_class->temp_hot_seconds < 0)
|
|
||||||
device_class->features |= FP_DEVICE_FEATURE_ALWAYS_ON;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -183,8 +177,8 @@ fpi_device_error_new (FpDeviceError error)
|
||||||
msg = "This device has been removed from the system.";
|
msg = "This device has been removed from the system.";
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case FP_DEVICE_ERROR_TOO_HOT:
|
case FP_DEVICE_ERROR_UNTRUSTED:
|
||||||
msg = "Device disabled to prevent overheating.";
|
msg = "Could not verify integrity of the device, it cannot be trusted!";
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
|
@ -498,11 +492,14 @@ gboolean
|
||||||
fpi_device_action_is_cancelled (FpDevice *device)
|
fpi_device_action_is_cancelled (FpDevice *device)
|
||||||
{
|
{
|
||||||
FpDevicePrivate *priv = fp_device_get_instance_private (device);
|
FpDevicePrivate *priv = fp_device_get_instance_private (device);
|
||||||
|
GCancellable *cancellable;
|
||||||
|
|
||||||
g_return_val_if_fail (FP_IS_DEVICE (device), TRUE);
|
g_return_val_if_fail (FP_IS_DEVICE (device), TRUE);
|
||||||
g_return_val_if_fail (priv->current_action != FPI_DEVICE_ACTION_NONE, TRUE);
|
g_return_val_if_fail (priv->current_action != FPI_DEVICE_ACTION_NONE, TRUE);
|
||||||
|
|
||||||
return g_cancellable_is_cancelled (priv->current_cancellable);
|
cancellable = g_task_get_cancellable (priv->current_task);
|
||||||
|
|
||||||
|
return g_cancellable_is_cancelled (cancellable);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -680,7 +677,7 @@ fpi_device_get_cancellable (FpDevice *device)
|
||||||
g_return_val_if_fail (FP_IS_DEVICE (device), NULL);
|
g_return_val_if_fail (FP_IS_DEVICE (device), NULL);
|
||||||
g_return_val_if_fail (priv->current_action != FPI_DEVICE_ACTION_NONE, NULL);
|
g_return_val_if_fail (priv->current_action != FPI_DEVICE_ACTION_NONE, NULL);
|
||||||
|
|
||||||
return priv->current_cancellable;
|
return g_task_get_cancellable (priv->current_task);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -809,120 +806,6 @@ fpi_device_action_error (FpDevice *device,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* fpi_device_critical_enter:
|
|
||||||
* @device: The #FpDevice
|
|
||||||
*
|
|
||||||
* Enter a critical section in the driver code where no outside calls from
|
|
||||||
* libfprint should happen. Drivers can already assume that everything
|
|
||||||
* happens from the same thread, however, that still allows e.g. the cancel
|
|
||||||
* vfunc to be called at any point in time.
|
|
||||||
*
|
|
||||||
* Using this kind of critical section, the driver can assume that libfprint
|
|
||||||
* will not forward any external requests to the driver for the time being.
|
|
||||||
* This is for example useful to prevent cancellation while the device is being
|
|
||||||
* set up. Or, said differently, using this feature means that the cancel
|
|
||||||
* handler is able to make more assumptions about the current state.
|
|
||||||
*
|
|
||||||
* Please note that the driver is not shielded from all external changes. For
|
|
||||||
* example the cancellable as returned by fpi_device_get_cancellable() will
|
|
||||||
* still change immediately.
|
|
||||||
*
|
|
||||||
* The driver may call this function multiple times, but must also ensure that
|
|
||||||
* fpi_device_critical_leave() is called an equal amount of times and that all
|
|
||||||
* critical sections are left before command completion.
|
|
||||||
*/
|
|
||||||
void
|
|
||||||
fpi_device_critical_enter (FpDevice *device)
|
|
||||||
{
|
|
||||||
FpDevicePrivate *priv = fp_device_get_instance_private (device);
|
|
||||||
|
|
||||||
g_return_if_fail (priv->current_action != FPI_DEVICE_ACTION_NONE);
|
|
||||||
|
|
||||||
priv->critical_section += 1;
|
|
||||||
|
|
||||||
/* Stop flushing events if that was previously queued. */
|
|
||||||
if (priv->critical_section_flush_source)
|
|
||||||
g_source_destroy (priv->critical_section_flush_source);
|
|
||||||
priv->critical_section_flush_source = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static gboolean
|
|
||||||
fpi_device_critical_section_flush_idle_cb (FpDevice *device)
|
|
||||||
{
|
|
||||||
FpDevicePrivate *priv = fp_device_get_instance_private (device);
|
|
||||||
FpDeviceClass *cls = FP_DEVICE_GET_CLASS (device);
|
|
||||||
|
|
||||||
if (priv->cancel_queued)
|
|
||||||
{
|
|
||||||
/* Cancellation must only happen if the driver is busy. */
|
|
||||||
if (priv->current_action != FPI_DEVICE_ACTION_NONE &&
|
|
||||||
priv->current_task_idle_return_source == NULL)
|
|
||||||
cls->cancel (device);
|
|
||||||
priv->cancel_queued = FALSE;
|
|
||||||
|
|
||||||
return G_SOURCE_CONTINUE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (priv->suspend_queued)
|
|
||||||
{
|
|
||||||
cls->suspend (device);
|
|
||||||
priv->suspend_queued = FALSE;
|
|
||||||
|
|
||||||
return G_SOURCE_CONTINUE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (priv->resume_queued)
|
|
||||||
{
|
|
||||||
cls->resume (device);
|
|
||||||
priv->resume_queued = FALSE;
|
|
||||||
|
|
||||||
return G_SOURCE_CONTINUE;
|
|
||||||
}
|
|
||||||
|
|
||||||
priv->critical_section_flush_source = NULL;
|
|
||||||
|
|
||||||
return G_SOURCE_REMOVE;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* fpi_device_critical_leave:
|
|
||||||
* @device: The #FpDevice
|
|
||||||
*
|
|
||||||
* Leave a critical section started by fpi_device_critical_enter().
|
|
||||||
*
|
|
||||||
* Once all critical sections have been left, libfprint will start flushing
|
|
||||||
* out the queued up requests. This is done from the mainloop and the driver
|
|
||||||
* is protected from reentrency issues.
|
|
||||||
*/
|
|
||||||
void
|
|
||||||
fpi_device_critical_leave (FpDevice *device)
|
|
||||||
{
|
|
||||||
FpDevicePrivate *priv = fp_device_get_instance_private (device);
|
|
||||||
|
|
||||||
g_return_if_fail (priv->current_action != FPI_DEVICE_ACTION_NONE);
|
|
||||||
g_return_if_fail (priv->critical_section);
|
|
||||||
|
|
||||||
priv->critical_section -= 1;
|
|
||||||
if (priv->critical_section)
|
|
||||||
return;
|
|
||||||
|
|
||||||
/* We left the critical section, make sure a flush is queued. */
|
|
||||||
if (priv->critical_section_flush_source)
|
|
||||||
return;
|
|
||||||
|
|
||||||
priv->critical_section_flush_source = g_idle_source_new ();
|
|
||||||
g_source_set_callback (priv->critical_section_flush_source,
|
|
||||||
(GSourceFunc) fpi_device_critical_section_flush_idle_cb,
|
|
||||||
device,
|
|
||||||
NULL);
|
|
||||||
g_source_set_name (priv->critical_section_flush_source,
|
|
||||||
"Flush libfprint driver critical section");
|
|
||||||
g_source_attach (priv->critical_section_flush_source,
|
|
||||||
g_task_get_context (priv->current_task));
|
|
||||||
g_source_unref (priv->critical_section_flush_source);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
clear_device_cancel_action (FpDevice *device)
|
clear_device_cancel_action (FpDevice *device)
|
||||||
{
|
{
|
||||||
|
@ -932,17 +815,10 @@ clear_device_cancel_action (FpDevice *device)
|
||||||
|
|
||||||
if (priv->current_cancellable_id)
|
if (priv->current_cancellable_id)
|
||||||
{
|
{
|
||||||
g_cancellable_disconnect (priv->current_cancellable,
|
g_cancellable_disconnect (g_task_get_cancellable (priv->current_task),
|
||||||
priv->current_cancellable_id);
|
priv->current_cancellable_id);
|
||||||
priv->current_cancellable_id = 0;
|
priv->current_cancellable_id = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (priv->current_task_cancellable_id)
|
|
||||||
{
|
|
||||||
g_cancellable_disconnect (g_task_get_cancellable (priv->current_task),
|
|
||||||
priv->current_task_cancellable_id);
|
|
||||||
priv->current_task_cancellable_id = 0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
typedef enum _FpDeviceTaskReturnType {
|
typedef enum _FpDeviceTaskReturnType {
|
||||||
|
@ -969,7 +845,6 @@ fp_device_task_return_in_idle_cb (gpointer user_data)
|
||||||
FpiDeviceAction action;
|
FpiDeviceAction action;
|
||||||
|
|
||||||
g_autoptr(GTask) task = NULL;
|
g_autoptr(GTask) task = NULL;
|
||||||
g_autoptr(GError) cancellation_reason = NULL;
|
|
||||||
|
|
||||||
|
|
||||||
action_str = g_enum_to_string (FPI_TYPE_DEVICE_ACTION, priv->current_action);
|
action_str = g_enum_to_string (FPI_TYPE_DEVICE_ACTION, priv->current_action);
|
||||||
|
@ -979,10 +854,6 @@ fp_device_task_return_in_idle_cb (gpointer user_data)
|
||||||
action = priv->current_action;
|
action = priv->current_action;
|
||||||
priv->current_action = FPI_DEVICE_ACTION_NONE;
|
priv->current_action = FPI_DEVICE_ACTION_NONE;
|
||||||
priv->current_task_idle_return_source = NULL;
|
priv->current_task_idle_return_source = NULL;
|
||||||
g_clear_object (&priv->current_cancellable);
|
|
||||||
cancellation_reason = g_steal_pointer (&priv->current_cancellation_reason);
|
|
||||||
|
|
||||||
fpi_device_update_temp (data->device, FALSE);
|
|
||||||
|
|
||||||
if (action == FPI_DEVICE_ACTION_OPEN &&
|
if (action == FPI_DEVICE_ACTION_OPEN &&
|
||||||
data->type != FP_DEVICE_TASK_RETURN_ERROR)
|
data->type != FP_DEVICE_TASK_RETURN_ERROR)
|
||||||
|
@ -999,8 +870,6 @@ fp_device_task_return_in_idle_cb (gpointer user_data)
|
||||||
g_object_notify (G_OBJECT (data->device), "open");
|
g_object_notify (G_OBJECT (data->device), "open");
|
||||||
}
|
}
|
||||||
|
|
||||||
/* TODO: Port/use the cancellation mechanism for device removal! */
|
|
||||||
|
|
||||||
/* Return FP_DEVICE_ERROR_REMOVED if the device is removed,
|
/* Return FP_DEVICE_ERROR_REMOVED if the device is removed,
|
||||||
* with the exception of a successful open, which is an odd corner case. */
|
* with the exception of a successful open, which is an odd corner case. */
|
||||||
if (priv->is_removed &&
|
if (priv->is_removed &&
|
||||||
|
@ -1036,18 +905,7 @@ fp_device_task_return_in_idle_cb (gpointer user_data)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case FP_DEVICE_TASK_RETURN_ERROR:
|
case FP_DEVICE_TASK_RETURN_ERROR:
|
||||||
/* Return internal cancellation reason instead if we have one.
|
|
||||||
* Note that an external cancellation always returns G_IO_ERROR_CANCELLED
|
|
||||||
*/
|
|
||||||
if (cancellation_reason)
|
|
||||||
{
|
|
||||||
g_task_set_task_data (task, NULL, NULL);
|
|
||||||
g_task_return_error (task, g_steal_pointer (&cancellation_reason));
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
g_task_return_error (task, g_steal_pointer (&data->result));
|
g_task_return_error (task, g_steal_pointer (&data->result));
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
|
@ -1550,183 +1408,6 @@ fpi_device_list_complete (FpDevice *device,
|
||||||
fpi_device_return_task_in_idle (device, FP_DEVICE_TASK_RETURN_ERROR, error);
|
fpi_device_return_task_in_idle (device, FP_DEVICE_TASK_RETURN_ERROR, error);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
|
||||||
fpi_device_configure_wakeup (FpDevice *device, gboolean enabled)
|
|
||||||
{
|
|
||||||
FpDevicePrivate *priv = fp_device_get_instance_private (device);
|
|
||||||
|
|
||||||
switch (priv->type)
|
|
||||||
{
|
|
||||||
case FP_DEVICE_TYPE_USB:
|
|
||||||
{
|
|
||||||
g_autoptr(GString) ports = NULL;
|
|
||||||
GUsbDevice *dev, *parent;
|
|
||||||
const char *wakeup_command = enabled ? "enabled" : "disabled";
|
|
||||||
guint8 bus, port;
|
|
||||||
g_autofree gchar *sysfs_wakeup = NULL;
|
|
||||||
g_autofree gchar *sysfs_persist = NULL;
|
|
||||||
gssize r;
|
|
||||||
int fd;
|
|
||||||
|
|
||||||
ports = g_string_new (NULL);
|
|
||||||
bus = g_usb_device_get_bus (priv->usb_device);
|
|
||||||
|
|
||||||
/* Walk up, skipping the root hub. */
|
|
||||||
dev = priv->usb_device;
|
|
||||||
while ((parent = g_usb_device_get_parent (dev)))
|
|
||||||
{
|
|
||||||
port = g_usb_device_get_port_number (dev);
|
|
||||||
g_string_prepend (ports, g_strdup_printf ("%d.", port));
|
|
||||||
dev = parent;
|
|
||||||
}
|
|
||||||
g_string_set_size (ports, ports->len - 1);
|
|
||||||
|
|
||||||
sysfs_wakeup = g_strdup_printf ("/sys/bus/usb/devices/%d-%s/power/wakeup", bus, ports->str);
|
|
||||||
fd = open (sysfs_wakeup, O_WRONLY);
|
|
||||||
|
|
||||||
if (fd < 0)
|
|
||||||
{
|
|
||||||
/* Wakeup not existing appears to be relatively normal. */
|
|
||||||
g_debug ("Failed to open %s", sysfs_wakeup);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
r = write (fd, wakeup_command, strlen (wakeup_command));
|
|
||||||
if (r < 0)
|
|
||||||
g_warning ("Could not configure wakeup to %s by writing %s", wakeup_command, sysfs_wakeup);
|
|
||||||
close (fd);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Persist means that the kernel tries to keep the USB device open
|
|
||||||
* in case it is "replugged" due to suspend.
|
|
||||||
* This is not helpful, as it will receive a reset and will be in a bad
|
|
||||||
* state. Instead, seeing an unplug and a new device makes more sense.
|
|
||||||
*/
|
|
||||||
sysfs_persist = g_strdup_printf ("/sys/bus/usb/devices/%d-%s/power/persist", bus, ports->str);
|
|
||||||
fd = open (sysfs_persist, O_WRONLY);
|
|
||||||
|
|
||||||
if (fd < 0)
|
|
||||||
{
|
|
||||||
g_warning ("Failed to open %s", sysfs_persist);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
r = write (fd, "0", 1);
|
|
||||||
if (r < 0)
|
|
||||||
g_message ("Could not disable USB persist by writing to %s", sysfs_persist);
|
|
||||||
close (fd);
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
case FP_DEVICE_TYPE_VIRTUAL:
|
|
||||||
case FP_DEVICE_TYPE_UDEV:
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
g_assert_not_reached ();
|
|
||||||
fpi_device_return_task_in_idle (device, FP_DEVICE_TASK_RETURN_ERROR,
|
|
||||||
fpi_device_error_new (FP_DEVICE_ERROR_GENERAL));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
fpi_device_suspend_completed (FpDevice *device)
|
|
||||||
{
|
|
||||||
FpDevicePrivate *priv = fp_device_get_instance_private (device);
|
|
||||||
|
|
||||||
/* We have an ongoing operation, allow the device to wake up the machine. */
|
|
||||||
if (priv->current_action != FPI_DEVICE_ACTION_NONE)
|
|
||||||
fpi_device_configure_wakeup (device, TRUE);
|
|
||||||
|
|
||||||
if (priv->critical_section)
|
|
||||||
g_warning ("Driver was in a critical section at suspend time. It likely deadlocked!");
|
|
||||||
|
|
||||||
if (priv->suspend_error)
|
|
||||||
g_task_return_error (g_steal_pointer (&priv->suspend_resume_task),
|
|
||||||
g_steal_pointer (&priv->suspend_error));
|
|
||||||
else
|
|
||||||
g_task_return_boolean (g_steal_pointer (&priv->suspend_resume_task), TRUE);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* fpi_device_suspend_complete:
|
|
||||||
* @device: The #FpDevice
|
|
||||||
* @error: The #GError or %NULL on success
|
|
||||||
*
|
|
||||||
* Finish a suspend request. Only return a %NULL error if suspend has been
|
|
||||||
* correctly configured and the current action as returned by
|
|
||||||
* fpi_device_get_current_action() will continue to run after resume.
|
|
||||||
*
|
|
||||||
* In all other cases an error must be returned. Should this happen, the
|
|
||||||
* current action will be cancelled before the error is forwarded to the
|
|
||||||
* application.
|
|
||||||
*
|
|
||||||
* It is recommended to set @error to #FP_ERROR_NOT_IMPLEMENTED.
|
|
||||||
*/
|
|
||||||
void
|
|
||||||
fpi_device_suspend_complete (FpDevice *device,
|
|
||||||
GError *error)
|
|
||||||
{
|
|
||||||
FpDevicePrivate *priv = fp_device_get_instance_private (device);
|
|
||||||
|
|
||||||
g_return_if_fail (FP_IS_DEVICE (device));
|
|
||||||
g_return_if_fail (priv->suspend_resume_task);
|
|
||||||
g_return_if_fail (priv->suspend_error == NULL);
|
|
||||||
|
|
||||||
priv->suspend_error = error;
|
|
||||||
priv->is_suspended = TRUE;
|
|
||||||
|
|
||||||
/* If there is no error, we have no running task, return immediately. */
|
|
||||||
if (error == NULL || !priv->current_task || g_task_get_completed (priv->current_task))
|
|
||||||
{
|
|
||||||
fpi_device_suspend_completed (device);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Wait for completion of the current task. */
|
|
||||||
g_signal_connect_object (priv->current_task,
|
|
||||||
"notify::completed",
|
|
||||||
G_CALLBACK (fpi_device_suspend_completed),
|
|
||||||
device,
|
|
||||||
G_CONNECT_SWAPPED);
|
|
||||||
|
|
||||||
/* And cancel any action that might be long-running. */
|
|
||||||
if (!priv->current_cancellation_reason)
|
|
||||||
priv->current_cancellation_reason = fpi_device_error_new_msg (FP_DEVICE_ERROR_BUSY,
|
|
||||||
"Cannot run while suspended.");
|
|
||||||
|
|
||||||
g_cancellable_cancel (priv->current_cancellable);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* fpi_device_resume_complete:
|
|
||||||
* @device: The #FpDevice
|
|
||||||
* @error: The #GError or %NULL on success
|
|
||||||
*
|
|
||||||
* Finish a resume request.
|
|
||||||
*/
|
|
||||||
void
|
|
||||||
fpi_device_resume_complete (FpDevice *device,
|
|
||||||
GError *error)
|
|
||||||
{
|
|
||||||
FpDevicePrivate *priv = fp_device_get_instance_private (device);
|
|
||||||
|
|
||||||
g_return_if_fail (FP_IS_DEVICE (device));
|
|
||||||
g_return_if_fail (priv->suspend_resume_task);
|
|
||||||
|
|
||||||
priv->is_suspended = FALSE;
|
|
||||||
fpi_device_configure_wakeup (device, FALSE);
|
|
||||||
|
|
||||||
if (error)
|
|
||||||
g_task_return_error (g_steal_pointer (&priv->suspend_resume_task), error);
|
|
||||||
else
|
|
||||||
g_task_return_boolean (g_steal_pointer (&priv->suspend_resume_task), TRUE);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* fpi_device_clear_storage_complete:
|
* fpi_device_clear_storage_complete:
|
||||||
* @device: The #FpDevice
|
* @device: The #FpDevice
|
||||||
|
@ -2008,128 +1689,3 @@ fpi_device_report_finger_status_changes (FpDevice *device,
|
||||||
|
|
||||||
return fpi_device_report_finger_status (device, finger_status);
|
return fpi_device_report_finger_status (device, finger_status);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
update_temp_timeout (FpDevice *device, gpointer user_data)
|
|
||||||
{
|
|
||||||
FpDevicePrivate *priv = fp_device_get_instance_private (device);
|
|
||||||
|
|
||||||
fpi_device_update_temp (device, priv->temp_last_active);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* fpi_device_update_temp:
|
|
||||||
* @device: The #FpDevice
|
|
||||||
* @is_active: Whether the device is now active
|
|
||||||
*
|
|
||||||
* Purely internal function to update the temperature. Also ensure that the
|
|
||||||
* state is updated once a threshold is reached.
|
|
||||||
*/
|
|
||||||
void
|
|
||||||
fpi_device_update_temp (FpDevice *device, gboolean is_active)
|
|
||||||
{
|
|
||||||
FpDevicePrivate *priv = fp_device_get_instance_private (device);
|
|
||||||
gint64 now = g_get_monotonic_time ();
|
|
||||||
gdouble passed_seconds;
|
|
||||||
gdouble alpha;
|
|
||||||
gdouble next_threshold;
|
|
||||||
gdouble old_ratio;
|
|
||||||
FpTemperature old_temp;
|
|
||||||
g_autofree char *old_temp_str = NULL;
|
|
||||||
g_autofree char *new_temp_str = NULL;
|
|
||||||
|
|
||||||
if (priv->temp_hot_seconds < 0)
|
|
||||||
{
|
|
||||||
g_debug ("Not updating temperature model, device can run continuously!");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
passed_seconds = (now - priv->temp_last_update) / 1e6;
|
|
||||||
old_ratio = priv->temp_current_ratio;
|
|
||||||
|
|
||||||
if (priv->temp_last_active)
|
|
||||||
{
|
|
||||||
alpha = exp (-passed_seconds / priv->temp_hot_seconds);
|
|
||||||
priv->temp_current_ratio = alpha * priv->temp_current_ratio + 1 - alpha;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
alpha = exp (-passed_seconds / priv->temp_cold_seconds);
|
|
||||||
priv->temp_current_ratio = alpha * priv->temp_current_ratio;
|
|
||||||
}
|
|
||||||
|
|
||||||
priv->temp_last_active = is_active;
|
|
||||||
priv->temp_last_update = now;
|
|
||||||
|
|
||||||
old_temp = priv->temp_current;
|
|
||||||
if (priv->temp_current_ratio < TEMP_COLD_THRESH)
|
|
||||||
{
|
|
||||||
priv->temp_current = FP_TEMPERATURE_COLD;
|
|
||||||
next_threshold = is_active ? TEMP_COLD_THRESH : -1.0;
|
|
||||||
}
|
|
||||||
else if (priv->temp_current_ratio < TEMP_HOT_WARM_THRESH)
|
|
||||||
{
|
|
||||||
priv->temp_current = FP_TEMPERATURE_WARM;
|
|
||||||
next_threshold = is_active ? TEMP_WARM_HOT_THRESH : TEMP_COLD_THRESH;
|
|
||||||
}
|
|
||||||
else if (priv->temp_current_ratio < TEMP_WARM_HOT_THRESH)
|
|
||||||
{
|
|
||||||
/* Keep HOT until we reach TEMP_HOT_WARM_THRESH */
|
|
||||||
if (priv->temp_current != FP_TEMPERATURE_HOT)
|
|
||||||
priv->temp_current = FP_TEMPERATURE_WARM;
|
|
||||||
|
|
||||||
next_threshold = is_active ? TEMP_WARM_HOT_THRESH : TEMP_HOT_WARM_THRESH;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
priv->temp_current = FP_TEMPERATURE_HOT;
|
|
||||||
next_threshold = is_active ? -1.0 : TEMP_HOT_WARM_THRESH;
|
|
||||||
}
|
|
||||||
|
|
||||||
old_temp_str = g_enum_to_string (FP_TYPE_TEMPERATURE, old_temp);
|
|
||||||
new_temp_str = g_enum_to_string (FP_TYPE_TEMPERATURE, priv->temp_current);
|
|
||||||
g_debug ("Updated temperature model after %0.2f seconds, ratio %0.2f -> %0.2f, active %d -> %d, %s -> %s",
|
|
||||||
passed_seconds,
|
|
||||||
old_ratio,
|
|
||||||
priv->temp_current_ratio,
|
|
||||||
priv->temp_last_active,
|
|
||||||
is_active,
|
|
||||||
old_temp_str,
|
|
||||||
new_temp_str);
|
|
||||||
|
|
||||||
if (priv->temp_current != old_temp)
|
|
||||||
g_object_notify (G_OBJECT (device), "temperature");
|
|
||||||
|
|
||||||
/* If the device is HOT, then do an internal cancellation of long running tasks. */
|
|
||||||
if (priv->temp_current == FP_TEMPERATURE_HOT)
|
|
||||||
{
|
|
||||||
if (priv->current_action == FPI_DEVICE_ACTION_ENROLL ||
|
|
||||||
priv->current_action == FPI_DEVICE_ACTION_VERIFY ||
|
|
||||||
priv->current_action == FPI_DEVICE_ACTION_IDENTIFY ||
|
|
||||||
priv->current_action == FPI_DEVICE_ACTION_CAPTURE)
|
|
||||||
{
|
|
||||||
if (!priv->current_cancellation_reason)
|
|
||||||
priv->current_cancellation_reason = fpi_device_error_new (FP_DEVICE_ERROR_TOO_HOT);
|
|
||||||
|
|
||||||
g_cancellable_cancel (priv->current_cancellable);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
g_clear_pointer (&priv->temp_timeout, g_source_destroy);
|
|
||||||
|
|
||||||
if (next_threshold < 0)
|
|
||||||
return;
|
|
||||||
|
|
||||||
/* Set passed_seconds to the time until the next update is needed */
|
|
||||||
if (is_active)
|
|
||||||
passed_seconds = -priv->temp_hot_seconds * log ((next_threshold - 1.0) / (priv->temp_current_ratio - 1.0));
|
|
||||||
else
|
|
||||||
passed_seconds = -priv->temp_cold_seconds * log (next_threshold / priv->temp_current_ratio);
|
|
||||||
|
|
||||||
passed_seconds += TEMP_DELAY_SECONDS;
|
|
||||||
|
|
||||||
priv->temp_timeout = fpi_device_add_timeout (device,
|
|
||||||
passed_seconds * 1000,
|
|
||||||
update_temp_timeout,
|
|
||||||
NULL, NULL);
|
|
||||||
}
|
|
||||||
|
|
|
@ -24,6 +24,8 @@
|
||||||
#include "fp-image.h"
|
#include "fp-image.h"
|
||||||
#include "fpi-print.h"
|
#include "fpi-print.h"
|
||||||
|
|
||||||
|
#include <config.h>
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* FpiDeviceUdevSubtype:
|
* FpiDeviceUdevSubtype:
|
||||||
* @FPI_DEVICE_UDEV_SUBTYPE_SPIDEV: The device requires an spidev node
|
* @FPI_DEVICE_UDEV_SUBTYPE_SPIDEV: The device requires an spidev node
|
||||||
|
@ -80,10 +82,6 @@ struct _FpIdEntry
|
||||||
* fpi_device_set_nr_enroll_stages() from @probe if this is dynamic.
|
* fpi_device_set_nr_enroll_stages() from @probe if this is dynamic.
|
||||||
* @scan_type: The scan type of supported devices; use
|
* @scan_type: The scan type of supported devices; use
|
||||||
* fpi_device_set_scan_type() from @probe if this is dynamic.
|
* fpi_device_set_scan_type() from @probe if this is dynamic.
|
||||||
* @temp_hot_seconds: Assumed time in seconds for the device to become too hot
|
|
||||||
* after being mostly cold. Set to -1 if the device can be always-on.
|
|
||||||
* @temp_cold_seconds: Assumed time in seconds for the device to be mostly cold
|
|
||||||
* after having been too hot to operate.
|
|
||||||
* @usb_discover: Class method to check whether a USB device is supported by
|
* @usb_discover: Class method to check whether a USB device is supported by
|
||||||
* the driver. Should return 0 if the device is unsupported and a positive
|
* the driver. Should return 0 if the device is unsupported and a positive
|
||||||
* score otherwise. The default score is 50 and the driver with the highest
|
* score otherwise. The default score is 50 and the driver with the highest
|
||||||
|
@ -106,10 +104,6 @@ struct _FpIdEntry
|
||||||
* @clear_storage: Delete all prints from the device
|
* @clear_storage: Delete all prints from the device
|
||||||
* @cancel: Called on cancellation, this is a convenience to not need to handle
|
* @cancel: Called on cancellation, this is a convenience to not need to handle
|
||||||
* the #GCancellable directly by using fpi_device_get_cancellable().
|
* the #GCancellable directly by using fpi_device_get_cancellable().
|
||||||
* @suspend: Called when an interactive action is running (ENROLL, VERIFY,
|
|
||||||
* IDENTIFY or CAPTURE) and the system is about to go into suspend.
|
|
||||||
* @resume: Called to resume an ongoing interactive action after the system has
|
|
||||||
* resumed from suspend.
|
|
||||||
*
|
*
|
||||||
* NOTE: If your driver is image based, then you should subclass #FpImageDevice
|
* NOTE: If your driver is image based, then you should subclass #FpImageDevice
|
||||||
* instead. #FpImageDevice based drivers use a different way of interacting
|
* instead. #FpImageDevice based drivers use a different way of interacting
|
||||||
|
@ -128,9 +122,6 @@ struct _FpIdEntry
|
||||||
* operation (i.e. any operation that requires capturing). It is entirely fine
|
* operation (i.e. any operation that requires capturing). It is entirely fine
|
||||||
* to ignore cancellation requests for short operations (e.g. open/close).
|
* to ignore cancellation requests for short operations (e.g. open/close).
|
||||||
*
|
*
|
||||||
* Note that @cancel, @suspend and @resume will not be called while the device
|
|
||||||
* is within a fpi_device_critical_enter()/fpi_device_critical_leave() block.
|
|
||||||
*
|
|
||||||
* This API is solely intended for drivers. It is purely internal and neither
|
* This API is solely intended for drivers. It is purely internal and neither
|
||||||
* API nor ABI stable.
|
* API nor ABI stable.
|
||||||
*/
|
*/
|
||||||
|
@ -151,10 +142,6 @@ struct _FpDeviceClass
|
||||||
gint nr_enroll_stages;
|
gint nr_enroll_stages;
|
||||||
FpScanType scan_type;
|
FpScanType scan_type;
|
||||||
|
|
||||||
/* Simple device temperature model constants */
|
|
||||||
gint32 temp_hot_seconds;
|
|
||||||
gint32 temp_cold_seconds;
|
|
||||||
|
|
||||||
/* Callbacks */
|
/* Callbacks */
|
||||||
gint (*usb_discover) (GUsbDevice *usb_device);
|
gint (*usb_discover) (GUsbDevice *usb_device);
|
||||||
void (*probe) (FpDevice *device);
|
void (*probe) (FpDevice *device);
|
||||||
|
@ -169,8 +156,6 @@ struct _FpDeviceClass
|
||||||
void (*clear_storage) (FpDevice * device);
|
void (*clear_storage) (FpDevice * device);
|
||||||
|
|
||||||
void (*cancel) (FpDevice *device);
|
void (*cancel) (FpDevice *device);
|
||||||
void (*suspend) (FpDevice *device);
|
|
||||||
void (*resume) (FpDevice *device);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
void fpi_device_class_auto_initialize_features (FpDeviceClass *device_class);
|
void fpi_device_class_auto_initialize_features (FpDeviceClass *device_class);
|
||||||
|
@ -271,9 +256,6 @@ void fpi_device_update_features (FpDevice *device,
|
||||||
void fpi_device_action_error (FpDevice *device,
|
void fpi_device_action_error (FpDevice *device,
|
||||||
GError *error);
|
GError *error);
|
||||||
|
|
||||||
void fpi_device_critical_enter (FpDevice *device);
|
|
||||||
void fpi_device_critical_leave (FpDevice *device);
|
|
||||||
|
|
||||||
void fpi_device_probe_complete (FpDevice *device,
|
void fpi_device_probe_complete (FpDevice *device,
|
||||||
const gchar *device_id,
|
const gchar *device_id,
|
||||||
const gchar *device_name,
|
const gchar *device_name,
|
||||||
|
@ -299,10 +281,6 @@ void fpi_device_list_complete (FpDevice *device,
|
||||||
GError *error);
|
GError *error);
|
||||||
void fpi_device_clear_storage_complete (FpDevice *device,
|
void fpi_device_clear_storage_complete (FpDevice *device,
|
||||||
GError *error);
|
GError *error);
|
||||||
void fpi_device_suspend_complete (FpDevice *device,
|
|
||||||
GError *error);
|
|
||||||
void fpi_device_resume_complete (FpDevice *device,
|
|
||||||
GError *error);
|
|
||||||
|
|
||||||
void fpi_device_enroll_progress (FpDevice *device,
|
void fpi_device_enroll_progress (FpDevice *device,
|
||||||
gint completed_stages,
|
gint completed_stages,
|
||||||
|
|
|
@ -11,11 +11,13 @@ G_BEGIN_DECLS
|
||||||
* @FPI_PRINT_UNDEFINED: Undefined type, this happens prior to enrollment
|
* @FPI_PRINT_UNDEFINED: Undefined type, this happens prior to enrollment
|
||||||
* @FPI_PRINT_RAW: A raw print where the data is directly compared
|
* @FPI_PRINT_RAW: A raw print where the data is directly compared
|
||||||
* @FPI_PRINT_NBIS: NBIS minutiae comparison
|
* @FPI_PRINT_NBIS: NBIS minutiae comparison
|
||||||
|
* @FPI_PRINT_SDCP: Print from an SDCP conforming device
|
||||||
*/
|
*/
|
||||||
typedef enum {
|
typedef enum {
|
||||||
FPI_PRINT_UNDEFINED = 0,
|
FPI_PRINT_UNDEFINED = 0,
|
||||||
FPI_PRINT_RAW,
|
FPI_PRINT_RAW,
|
||||||
FPI_PRINT_NBIS,
|
FPI_PRINT_NBIS,
|
||||||
|
FPI_PRINT_SDCP,
|
||||||
} FpiPrintType;
|
} FpiPrintType;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
1469
libfprint/fpi-sdcp-device.c
Normal file
1469
libfprint/fpi-sdcp-device.c
Normal file
File diff suppressed because it is too large
Load diff
142
libfprint/fpi-sdcp-device.h
Normal file
142
libfprint/fpi-sdcp-device.h
Normal file
|
@ -0,0 +1,142 @@
|
||||||
|
/*
|
||||||
|
* FpSdcpDevice - A base class for SDCP enabled devices
|
||||||
|
* Copyright (C) 2020 Benjamin Berg <bberg@redhat.com>
|
||||||
|
*
|
||||||
|
* This library is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2.1 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This library is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this library; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <glib-2.0/glib.h>
|
||||||
|
#include "fpi-device.h"
|
||||||
|
#include "fp-sdcp-device.h"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* FpiSdcpClaim:
|
||||||
|
* @cert_m: The per-model ECDSA certificate (x509 ASN.1 DER encoded)
|
||||||
|
* @pk_d: The device public key (65 bytes)
|
||||||
|
* @pk_f: The firmware public key (65 bytes)
|
||||||
|
* @h_f: The firmware hash
|
||||||
|
* @s_m: Signature over @pk_d using the per-model private key (64 bytes)
|
||||||
|
* @s_d: Signature over h_f and pk_f using the device private key (64 bytes)
|
||||||
|
*
|
||||||
|
* Structure to hold the claim as produced by the device during a secure
|
||||||
|
* connect. See the SDCP specification for more details.
|
||||||
|
*
|
||||||
|
* Note all of these may simply be memory views into a larger #GBytes created
|
||||||
|
* using g_bytes_new_from_bytes().
|
||||||
|
*/
|
||||||
|
struct _FpiSdcpClaim
|
||||||
|
{
|
||||||
|
/*< public >*/
|
||||||
|
GBytes *cert_m;
|
||||||
|
GBytes *pk_d;
|
||||||
|
GBytes *pk_f;
|
||||||
|
GBytes *h_f;
|
||||||
|
GBytes *s_m;
|
||||||
|
GBytes *s_d;
|
||||||
|
};
|
||||||
|
typedef struct _FpiSdcpClaim FpiSdcpClaim;
|
||||||
|
|
||||||
|
GType fpi_sdcp_claim_get_type (void) G_GNUC_CONST;
|
||||||
|
FpiSdcpClaim *fpi_sdcp_claim_new (void);
|
||||||
|
FpiSdcpClaim *fpi_sdcp_claim_copy (FpiSdcpClaim *other);
|
||||||
|
void fpi_sdcp_claim_free (FpiSdcpClaim *claim);
|
||||||
|
|
||||||
|
G_DEFINE_AUTOPTR_CLEANUP_FUNC (FpiSdcpClaim, fpi_sdcp_claim_free)
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* FpSdcpDeviceClass:
|
||||||
|
* @connect: Establish SDCP connection. Similar to open in #FpDeviceClass
|
||||||
|
* but called connect to mirror the SDCP specification.
|
||||||
|
* @reconnect: Perform a faster reconnect. Drivers do not need to provide this
|
||||||
|
* function. If reconnect fails, then a normal connect will be tried.
|
||||||
|
* @enroll_begin: Start the enrollment procedure. In the absence of an error,
|
||||||
|
* the driver must call fpi_sdcp_device_enroll_set_nonce() at any point. It
|
||||||
|
* also must report enrollment progress using fpi_device_enroll_progress().
|
||||||
|
* It should also store available metadata about the print in device memory.
|
||||||
|
* The operation is completed with fpi_sdcp_device_enroll_ready().
|
||||||
|
* @enroll_commit: Complete the enrollment procedure. This commits the newly
|
||||||
|
* enrolled print to the device memory. Will only be called if enroll_begin
|
||||||
|
* succeeded. The passed id may be %NULL, in that case the driver must
|
||||||
|
* abort the enrollment process. id is owned by the base class and remains
|
||||||
|
* valid throughout the operation.
|
||||||
|
* @identify: Start identification process. On completion, the driver must call
|
||||||
|
* fpi_sdcp_device_identify_complete(). To request the user to retry the
|
||||||
|
* fpi_sdcp_device_identify_retry() function is used.
|
||||||
|
*
|
||||||
|
*
|
||||||
|
* These are the main entry points for drivers implementing SDCP.
|
||||||
|
*
|
||||||
|
* Drivers *must* eventually call the corresponding function to finish the
|
||||||
|
* operation.
|
||||||
|
*
|
||||||
|
* XXX: Is the use of fpi_device_action_error() acceptable?
|
||||||
|
*
|
||||||
|
* Drivers *must* also handle cancellation properly for any long running
|
||||||
|
* operation (i.e. any operation that requires capturing). It is entirely fine
|
||||||
|
* to ignore cancellation requests for short operations (e.g. open/close).
|
||||||
|
*
|
||||||
|
* This API is solely intended for drivers. It is purely internal and neither
|
||||||
|
* API nor ABI stable.
|
||||||
|
*/
|
||||||
|
struct _FpSdcpDeviceClass
|
||||||
|
{
|
||||||
|
FpDeviceClass parent_class;
|
||||||
|
|
||||||
|
void (*connect) (FpSdcpDevice *dev);
|
||||||
|
void (*reconnect) (FpSdcpDevice *dev);
|
||||||
|
void (*close) (FpSdcpDevice *dev);
|
||||||
|
void (*enroll_begin) (FpSdcpDevice *dev);
|
||||||
|
void (*enroll_commit) (FpSdcpDevice *dev,
|
||||||
|
GBytes *id);
|
||||||
|
void (*identify) (FpSdcpDevice *dev);
|
||||||
|
};
|
||||||
|
|
||||||
|
void fpi_sdcp_device_set_intermediat_cas (FpSdcpDevice *self,
|
||||||
|
GBytes *ca_1,
|
||||||
|
GBytes *ca_2);
|
||||||
|
|
||||||
|
void fpi_sdcp_device_get_connect_data (FpSdcpDevice *self,
|
||||||
|
GBytes **r_h,
|
||||||
|
GBytes **pk_h);
|
||||||
|
void fpi_sdcp_device_connect_complete (FpSdcpDevice *self,
|
||||||
|
GBytes *r_d,
|
||||||
|
FpiSdcpClaim *claim,
|
||||||
|
GBytes *mac,
|
||||||
|
GError *error);
|
||||||
|
|
||||||
|
void fpi_sdcp_device_get_reconnect_data (FpSdcpDevice *self,
|
||||||
|
GBytes **r_h);
|
||||||
|
void fpi_sdcp_device_reconnect_complete (FpSdcpDevice *self,
|
||||||
|
GBytes *mac,
|
||||||
|
GError *error);
|
||||||
|
|
||||||
|
void fpi_sdcp_device_enroll_set_nonce (FpSdcpDevice *self,
|
||||||
|
GBytes *nonce);
|
||||||
|
void fpi_sdcp_device_enroll_ready (FpSdcpDevice *self,
|
||||||
|
GError *error);
|
||||||
|
void fpi_sdcp_device_enroll_commit_complete (FpSdcpDevice *self,
|
||||||
|
GError *error);
|
||||||
|
|
||||||
|
void fpi_sdcp_device_get_identify_data (FpSdcpDevice *self,
|
||||||
|
GBytes **nonce);
|
||||||
|
void fpi_sdcp_device_identify_retry (FpSdcpDevice *self,
|
||||||
|
GError *error);
|
||||||
|
void fpi_sdcp_device_identify_complete (FpSdcpDevice *self,
|
||||||
|
GBytes *id,
|
||||||
|
GBytes *mac,
|
||||||
|
GError *error);
|
|
@ -323,7 +323,7 @@ transfer_chunk (FpiSpiTransfer *transfer, gsize full_length, gsize *transferred)
|
||||||
{
|
{
|
||||||
if (skip < transfer->length_wr && len < block_size)
|
if (skip < transfer->length_wr && len < block_size)
|
||||||
{
|
{
|
||||||
xfer[transfers].tx_buf = (gsize) transfer->buffer_wr + skip;
|
xfer[transfers].tx_buf = (guint64) transfer->buffer_wr + skip;
|
||||||
xfer[transfers].len = MIN (block_size, transfer->length_wr - skip);
|
xfer[transfers].len = MIN (block_size, transfer->length_wr - skip);
|
||||||
|
|
||||||
len += xfer[transfers].len;
|
len += xfer[transfers].len;
|
||||||
|
@ -340,7 +340,7 @@ transfer_chunk (FpiSpiTransfer *transfer, gsize full_length, gsize *transferred)
|
||||||
{
|
{
|
||||||
if (skip < transfer->length_rd && len < block_size)
|
if (skip < transfer->length_rd && len < block_size)
|
||||||
{
|
{
|
||||||
xfer[transfers].rx_buf = (gsize) transfer->buffer_rd + skip;
|
xfer[transfers].rx_buf = (guint64) transfer->buffer_rd + skip;
|
||||||
xfer[transfers].len = MIN (block_size, transfer->length_rd - skip);
|
xfer[transfers].len = MIN (block_size, transfer->length_rd - skip);
|
||||||
|
|
||||||
len += xfer[transfers].len;
|
len += xfer[transfers].len;
|
||||||
|
|
|
@ -49,7 +49,6 @@ static const FpIdEntry whitelist_id_table[] = {
|
||||||
{ .vid = 0x06cb, .pid = 0x00cb },
|
{ .vid = 0x06cb, .pid = 0x00cb },
|
||||||
{ .vid = 0x06cb, .pid = 0x00d8 },
|
{ .vid = 0x06cb, .pid = 0x00d8 },
|
||||||
{ .vid = 0x06cb, .pid = 0x00da },
|
{ .vid = 0x06cb, .pid = 0x00da },
|
||||||
{ .vid = 0x06cb, .pid = 0x00e7 },
|
|
||||||
{ .vid = 0x06cb, .pid = 0x00e9 },
|
{ .vid = 0x06cb, .pid = 0x00e9 },
|
||||||
{ .vid = 0x0a5c, .pid = 0x5801 },
|
{ .vid = 0x0a5c, .pid = 0x5801 },
|
||||||
{ .vid = 0x0a5c, .pid = 0x5805 },
|
{ .vid = 0x0a5c, .pid = 0x5805 },
|
||||||
|
@ -58,7 +57,6 @@ static const FpIdEntry whitelist_id_table[] = {
|
||||||
{ .vid = 0x0a5c, .pid = 0x5841 },
|
{ .vid = 0x0a5c, .pid = 0x5841 },
|
||||||
{ .vid = 0x0a5c, .pid = 0x5842 },
|
{ .vid = 0x0a5c, .pid = 0x5842 },
|
||||||
{ .vid = 0x0a5c, .pid = 0x5843 },
|
{ .vid = 0x0a5c, .pid = 0x5843 },
|
||||||
{ .vid = 0x0a5c, .pid = 0x5844 },
|
|
||||||
{ .vid = 0x0a5c, .pid = 0x5845 },
|
{ .vid = 0x0a5c, .pid = 0x5845 },
|
||||||
{ .vid = 0x10a5, .pid = 0x0007 },
|
{ .vid = 0x10a5, .pid = 0x0007 },
|
||||||
{ .vid = 0x1188, .pid = 0x9545 },
|
{ .vid = 0x1188, .pid = 0x9545 },
|
||||||
|
@ -77,6 +75,7 @@ static const FpIdEntry whitelist_id_table[] = {
|
||||||
{ .vid = 0x1491, .pid = 0x0088 },
|
{ .vid = 0x1491, .pid = 0x0088 },
|
||||||
{ .vid = 0x16d1, .pid = 0x1027 },
|
{ .vid = 0x16d1, .pid = 0x1027 },
|
||||||
{ .vid = 0x1c7a, .pid = 0x0300 },
|
{ .vid = 0x1c7a, .pid = 0x0300 },
|
||||||
|
{ .vid = 0x1c7a, .pid = 0x0570 },
|
||||||
{ .vid = 0x1c7a, .pid = 0x0575 },
|
{ .vid = 0x1c7a, .pid = 0x0575 },
|
||||||
{ .vid = 0x27c6, .pid = 0x5042 },
|
{ .vid = 0x27c6, .pid = 0x5042 },
|
||||||
{ .vid = 0x27c6, .pid = 0x5110 },
|
{ .vid = 0x27c6, .pid = 0x5110 },
|
||||||
|
@ -167,10 +166,7 @@ print_driver (const FpDeviceClass *cls)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (num_printed > 0)
|
if (num_printed > 0)
|
||||||
{
|
|
||||||
g_print (" ID_AUTOSUSPEND=1\n");
|
g_print (" ID_AUTOSUSPEND=1\n");
|
||||||
g_print (" ID_PERSIST=0\n");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
|
|
|
@ -117,14 +117,10 @@ driver_sources = {
|
||||||
[ 'drivers/upektc_img.c', 'drivers/upek_proto.c' ],
|
[ 'drivers/upektc_img.c', 'drivers/upek_proto.c' ],
|
||||||
'etes603' :
|
'etes603' :
|
||||||
[ 'drivers/etes603.c' ],
|
[ 'drivers/etes603.c' ],
|
||||||
'egis0570' :
|
|
||||||
[ 'drivers/egis0570.c' ],
|
|
||||||
'vfs0050' :
|
'vfs0050' :
|
||||||
[ 'drivers/vfs0050.c' ],
|
[ 'drivers/vfs0050.c' ],
|
||||||
'elan' :
|
'elan' :
|
||||||
[ 'drivers/elan.c' ],
|
[ 'drivers/elan.c' ],
|
||||||
'elanmoc' :
|
|
||||||
[ 'drivers/elanmoc/elanmoc.c' ],
|
|
||||||
'elanspi' :
|
'elanspi' :
|
||||||
[ 'drivers/elanspi.c' ],
|
[ 'drivers/elanspi.c' ],
|
||||||
'nb1010' :
|
'nb1010' :
|
||||||
|
@ -135,6 +131,8 @@ driver_sources = {
|
||||||
[ 'drivers/virtual-device.c' ],
|
[ 'drivers/virtual-device.c' ],
|
||||||
'virtual_device_storage' :
|
'virtual_device_storage' :
|
||||||
[ 'drivers/virtual-device-storage.c' ],
|
[ 'drivers/virtual-device-storage.c' ],
|
||||||
|
'virtual_sdcp' :
|
||||||
|
[ 'drivers/virtual-sdcp.c' ],
|
||||||
'synaptics' :
|
'synaptics' :
|
||||||
[ 'drivers/synaptics/synaptics.c', 'drivers/synaptics/bmkt_message.c' ],
|
[ 'drivers/synaptics/synaptics.c', 'drivers/synaptics/bmkt_message.c' ],
|
||||||
'goodixmoc' :
|
'goodixmoc' :
|
||||||
|
@ -142,6 +140,8 @@ driver_sources = {
|
||||||
}
|
}
|
||||||
|
|
||||||
helper_sources = {
|
helper_sources = {
|
||||||
|
'sdcp' :
|
||||||
|
[ 'fp-sdcp-device.c', 'fpi-sdcp-device.c' ],
|
||||||
'aeslib' :
|
'aeslib' :
|
||||||
[ 'drivers/aeslib.c' ],
|
[ 'drivers/aeslib.c' ],
|
||||||
'aesx660' :
|
'aesx660' :
|
||||||
|
|
14
meson.build
14
meson.build
|
@ -1,5 +1,5 @@
|
||||||
project('libfprint', [ 'c', 'cpp' ],
|
project('libfprint', [ 'c', 'cpp' ],
|
||||||
version: '1.94.0',
|
version: '1.92.0',
|
||||||
license: 'LGPLv2.1+',
|
license: 'LGPLv2.1+',
|
||||||
default_options: [
|
default_options: [
|
||||||
'buildtype=debugoptimized',
|
'buildtype=debugoptimized',
|
||||||
|
@ -96,8 +96,8 @@ virtual_drivers = [
|
||||||
'virtual_image',
|
'virtual_image',
|
||||||
'virtual_device',
|
'virtual_device',
|
||||||
'virtual_device_storage',
|
'virtual_device_storage',
|
||||||
|
'virtual_sdcp',
|
||||||
]
|
]
|
||||||
|
|
||||||
default_drivers = [
|
default_drivers = [
|
||||||
'upektc_img',
|
'upektc_img',
|
||||||
'vfs5011',
|
'vfs5011',
|
||||||
|
@ -113,11 +113,9 @@ default_drivers = [
|
||||||
'vfs301',
|
'vfs301',
|
||||||
'vfs0050',
|
'vfs0050',
|
||||||
'etes603',
|
'etes603',
|
||||||
'egis0570',
|
|
||||||
'vcom5s',
|
'vcom5s',
|
||||||
'synaptics',
|
'synaptics',
|
||||||
'elan',
|
'elan',
|
||||||
'elanmoc',
|
|
||||||
'uru4000',
|
'uru4000',
|
||||||
'upektc',
|
'upektc',
|
||||||
'upeksonly',
|
'upeksonly',
|
||||||
|
@ -159,6 +157,7 @@ driver_helper_mapping = {
|
||||||
'virtual_image' : [ 'virtual' ],
|
'virtual_image' : [ 'virtual' ],
|
||||||
'virtual_device' : [ 'virtual' ],
|
'virtual_device' : [ 'virtual' ],
|
||||||
'virtual_device_storage' : [ 'virtual' ],
|
'virtual_device_storage' : [ 'virtual' ],
|
||||||
|
'virtual_sdcp' : [ 'sdcp' ],
|
||||||
}
|
}
|
||||||
|
|
||||||
driver_helpers = []
|
driver_helpers = []
|
||||||
|
@ -217,6 +216,13 @@ foreach i : driver_helpers
|
||||||
error('nss is required for @0@ and possibly others'.format(driver))
|
error('nss is required for @0@ and possibly others'.format(driver))
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
optional_deps += nss_dep
|
||||||
|
elif i == 'sdcp'
|
||||||
|
nss_dep = dependency('nss', version: '>=3.55', required: false)
|
||||||
|
if not nss_dep.found()
|
||||||
|
error('nss >=3.55 is required for SDCP support (@0@ and possibly others)'.format(driver))
|
||||||
|
endif
|
||||||
|
|
||||||
optional_deps += nss_dep
|
optional_deps += nss_dep
|
||||||
elif i == 'udev'
|
elif i == 'udev'
|
||||||
install_udev_rules = true
|
install_udev_rules = true
|
||||||
|
|
|
@ -30,3 +30,8 @@ option('doc',
|
||||||
description: 'Whether to build the API documentation',
|
description: 'Whether to build the API documentation',
|
||||||
type: 'boolean',
|
type: 'boolean',
|
||||||
value: true)
|
value: true)
|
||||||
|
|
||||||
|
option('sdcp_virt_binary',
|
||||||
|
description: 'Path to virtual SDCP test binary, please refer to CI for more details.',
|
||||||
|
type: 'string',
|
||||||
|
value: '')
|
||||||
|
|
|
@ -15,23 +15,57 @@ script, capture it and store the capture to `custom.pcapng`.
|
||||||
-----------------------
|
-----------------------
|
||||||
A new 'capture' test is created by means of `capture.py` script:
|
A new 'capture' test is created by means of `capture.py` script:
|
||||||
|
|
||||||
1. Make sure that libfprint is built with support for the device driver
|
1. Create (if needed) a directory for the driver under `tests`
|
||||||
that you want to capture a test case for.
|
directory:
|
||||||
|
|
||||||
2. From the build directory, run tests/create-driver-test.py as root. Note
|
`mkdir DRIVER`
|
||||||
that if you're capturing data for a driver which already has a test case
|
|
||||||
but the hardware is slightly different, you might want to pass a variant
|
|
||||||
name as a command-line options, for example:
|
|
||||||
```sh
|
|
||||||
$ sudo tests/create-driver-test.py driver [variant]
|
|
||||||
```
|
|
||||||
|
|
||||||
3. If the capture is not successful, run the tool again to start another capture.
|
Note that the name must be the exact name of the libfprint driver,
|
||||||
|
or the exact name of the driver followed by a `-` and a unique identifier
|
||||||
|
of your choosing.
|
||||||
|
|
||||||
4. Add driver test name to `drivers_tests` in the `meson.build`, as instructed,
|
2. Prepare your execution environment.
|
||||||
and change the ownership of the just-created test directory in the source.
|
|
||||||
|
|
||||||
5. Check whether `meson test` passes with this new test.
|
In the next step a working and up to date libfprint is needed. This can be
|
||||||
|
achieved by installing it into your system. Alternatively, you can set
|
||||||
|
the following environment variables to run a local build:
|
||||||
|
- `export LD_PRELOAD=<meson-build-dir>/libfprint/libfprint-2.so`
|
||||||
|
- `export GI_TYPELIB_PATH=<meson-build-dir>/libfprint`
|
||||||
|
|
||||||
|
Also, sometimes the driver must be adapted to the emulated environment
|
||||||
|
(mainly if it uses random numbers, see `synaptics.c` for an example).
|
||||||
|
Set the following environment variable to enable this adaptation:
|
||||||
|
- `export FP_DEVICE_EMULATION=1`
|
||||||
|
|
||||||
|
Run the next steps in the same terminal.
|
||||||
|
|
||||||
|
3. Find the real USB fingerprint device with `lsusb`, e.g.:
|
||||||
|
|
||||||
|
`Bus 001 Device 005: ID 138a:0090 Validity Sensors, Inc. VFS7500 Touch Fingerprint Sensor`
|
||||||
|
|
||||||
|
The following USB device is used in the example above:
|
||||||
|
`/dev/bus/usb/001/005`.
|
||||||
|
|
||||||
|
For the following commands, it is assumed that the user that's
|
||||||
|
running the commands has full access to the device node, whether
|
||||||
|
by running the commands as `root`, or changing the permissions for
|
||||||
|
that device node.
|
||||||
|
|
||||||
|
4. Record information about this device:
|
||||||
|
|
||||||
|
`umockdev-record /dev/bus/usb/001/005 > DRIVER/device`
|
||||||
|
|
||||||
|
5. Record interaction of `capture.py` (or other test) with the device. To do
|
||||||
|
so, start wireshark and record `usbmonX` (where X is the bus number). Then
|
||||||
|
run the test script:
|
||||||
|
|
||||||
|
`python3 ./capture.py DRIVER/capture.png`
|
||||||
|
|
||||||
|
Save the wireshark recording as `capture.pcapng`. The command will create
|
||||||
|
`capture.png`.
|
||||||
|
|
||||||
|
6. Add driver's name to `drivers_tests` in the `meson.build`.
|
||||||
|
7. Check whether everything works as expected.
|
||||||
|
|
||||||
**Note.** To avoid submitting a real fingerprint, the side of finger,
|
**Note.** To avoid submitting a real fingerprint, the side of finger,
|
||||||
arm, or anything else producing an image with the device can be used.
|
arm, or anything else producing an image with the device can be used.
|
||||||
|
|
Binary file not shown.
Binary file not shown.
Before Width: | Height: | Size: 5.1 KiB |
|
@ -1,223 +0,0 @@
|
||||||
P: /devices/pci0000:00/0000:00:14.0/usb1/1-10
|
|
||||||
N: bus/usb/001/044=12011001FFFFFF08FF08802523060001000109022000010100A0320904000002FFFFFF000705810220000007050202080000
|
|
||||||
E: DEVNAME=/dev/bus/usb/001/044
|
|
||||||
E: DEVTYPE=usb_device
|
|
||||||
E: DRIVER=usb
|
|
||||||
E: PRODUCT=8ff/2580/623
|
|
||||||
E: TYPE=255/255/255
|
|
||||||
E: BUSNUM=001
|
|
||||||
E: DEVNUM=044
|
|
||||||
E: MAJOR=189
|
|
||||||
E: MINOR=43
|
|
||||||
E: SUBSYSTEM=usb
|
|
||||||
E: ID_VENDOR=08ff
|
|
||||||
E: ID_VENDOR_ENC=08ff
|
|
||||||
E: ID_VENDOR_ID=08ff
|
|
||||||
E: ID_MODEL=Fingerprint_Sensor
|
|
||||||
E: ID_MODEL_ENC=Fingerprint\x20Sensor
|
|
||||||
E: ID_MODEL_ID=2580
|
|
||||||
E: ID_REVISION=0623
|
|
||||||
E: ID_SERIAL=08ff_Fingerprint_Sensor
|
|
||||||
E: ID_BUS=usb
|
|
||||||
E: ID_USB_INTERFACES=:ffffff:
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=AuthenTec, Inc.
|
|
||||||
E: ID_AUTOSUSPEND=1
|
|
||||||
E: ID_MODEL_FROM_DATABASE=AES2501 Fingerprint Sensor
|
|
||||||
E: ID_PATH=pci-0000:00:14.0-usb-0:10
|
|
||||||
E: ID_PATH_TAG=pci-0000_00_14_0-usb-0_10
|
|
||||||
E: LIBFPRINT_DRIVER=AuthenTec AES2501
|
|
||||||
A: authorized=1\n
|
|
||||||
A: avoid_reset_quirk=0\n
|
|
||||||
A: bConfigurationValue=1\n
|
|
||||||
A: bDeviceClass=ff\n
|
|
||||||
A: bDeviceProtocol=ff\n
|
|
||||||
A: bDeviceSubClass=ff\n
|
|
||||||
A: bMaxPacketSize0=8\n
|
|
||||||
A: bMaxPower=100mA\n
|
|
||||||
A: bNumConfigurations=1\n
|
|
||||||
A: bNumInterfaces= 1\n
|
|
||||||
A: bcdDevice=0623\n
|
|
||||||
A: bmAttributes=a0\n
|
|
||||||
A: busnum=1\n
|
|
||||||
A: configuration=
|
|
||||||
H: descriptors=12011001FFFFFF08FF08802523060001000109022000010100A0320904000002FFFFFF000705810220000007050202080000
|
|
||||||
A: dev=189:43\n
|
|
||||||
A: devnum=44\n
|
|
||||||
A: devpath=10\n
|
|
||||||
L: driver=../../../../../bus/usb/drivers/usb
|
|
||||||
L: firmware_node=../../../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:4c/device:4d/device:57
|
|
||||||
A: idProduct=2580\n
|
|
||||||
A: idVendor=08ff\n
|
|
||||||
A: ltm_capable=no\n
|
|
||||||
A: maxchild=0\n
|
|
||||||
L: port=../1-0:1.0/usb1-port10
|
|
||||||
A: power/active_duration=10573\n
|
|
||||||
A: power/autosuspend=2\n
|
|
||||||
A: power/autosuspend_delay_ms=2000\n
|
|
||||||
A: power/connected_duration=23441\n
|
|
||||||
A: power/control=auto\n
|
|
||||||
A: power/level=auto\n
|
|
||||||
A: power/persist=0\n
|
|
||||||
A: power/runtime_active_time=10430\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=12771\n
|
|
||||||
A: power/wakeup=disabled\n
|
|
||||||
A: power/wakeup_abort_count=\n
|
|
||||||
A: power/wakeup_active=\n
|
|
||||||
A: power/wakeup_active_count=\n
|
|
||||||
A: power/wakeup_count=\n
|
|
||||||
A: power/wakeup_expire_count=\n
|
|
||||||
A: power/wakeup_last_time_ms=\n
|
|
||||||
A: power/wakeup_max_time_ms=\n
|
|
||||||
A: power/wakeup_total_time_ms=\n
|
|
||||||
A: product=Fingerprint Sensor\n
|
|
||||||
A: quirks=0x0\n
|
|
||||||
A: removable=removable\n
|
|
||||||
A: rx_lanes=1\n
|
|
||||||
A: speed=12\n
|
|
||||||
A: tx_lanes=1\n
|
|
||||||
A: urbnum=13241\n
|
|
||||||
A: version= 1.10\n
|
|
||||||
|
|
||||||
P: /devices/pci0000:00/0000:00:14.0/usb1
|
|
||||||
N: bus/usb/001/001=12010002090001406B1D020013050302010109021900010100E0000904000001090000000705810304000C
|
|
||||||
E: DEVNAME=/dev/bus/usb/001/001
|
|
||||||
E: DEVTYPE=usb_device
|
|
||||||
E: DRIVER=usb
|
|
||||||
E: PRODUCT=1d6b/2/513
|
|
||||||
E: TYPE=9/0/1
|
|
||||||
E: BUSNUM=001
|
|
||||||
E: DEVNUM=001
|
|
||||||
E: MAJOR=189
|
|
||||||
E: MINOR=0
|
|
||||||
E: SUBSYSTEM=usb
|
|
||||||
E: ID_VENDOR=Linux_5.13.12-200.fc34.x86_64_xhci-hcd
|
|
||||||
E: ID_VENDOR_ENC=Linux\x205.13.12-200.fc34.x86_64\x20xhci-hcd
|
|
||||||
E: ID_VENDOR_ID=1d6b
|
|
||||||
E: ID_MODEL=xHCI_Host_Controller
|
|
||||||
E: ID_MODEL_ENC=xHCI\x20Host\x20Controller
|
|
||||||
E: ID_MODEL_ID=0002
|
|
||||||
E: ID_REVISION=0513
|
|
||||||
E: ID_SERIAL=Linux_5.13.12-200.fc34.x86_64_xhci-hcd_xHCI_Host_Controller_0000:00:14.0
|
|
||||||
E: ID_SERIAL_SHORT=0000:00:14.0
|
|
||||||
E: ID_BUS=usb
|
|
||||||
E: ID_USB_INTERFACES=:090000:
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Linux Foundation
|
|
||||||
E: ID_AUTOSUSPEND=1
|
|
||||||
E: ID_MODEL_FROM_DATABASE=2.0 root hub
|
|
||||||
E: ID_PATH=pci-0000:00:14.0
|
|
||||||
E: ID_PATH_TAG=pci-0000_00_14_0
|
|
||||||
E: ID_FOR_SEAT=usb-pci-0000_00_14_0
|
|
||||||
E: TAGS=:seat:
|
|
||||||
E: CURRENT_TAGS=:seat:
|
|
||||||
A: authorized=1\n
|
|
||||||
A: authorized_default=1\n
|
|
||||||
A: avoid_reset_quirk=0\n
|
|
||||||
A: bConfigurationValue=1\n
|
|
||||||
A: bDeviceClass=09\n
|
|
||||||
A: bDeviceProtocol=01\n
|
|
||||||
A: bDeviceSubClass=00\n
|
|
||||||
A: bMaxPacketSize0=64\n
|
|
||||||
A: bMaxPower=0mA\n
|
|
||||||
A: bNumConfigurations=1\n
|
|
||||||
A: bNumInterfaces= 1\n
|
|
||||||
A: bcdDevice=0513\n
|
|
||||||
A: bmAttributes=e0\n
|
|
||||||
A: busnum=1\n
|
|
||||||
A: configuration=
|
|
||||||
H: descriptors=12010002090001406B1D020013050302010109021900010100E0000904000001090000000705810304000C
|
|
||||||
A: dev=189:0\n
|
|
||||||
A: devnum=1\n
|
|
||||||
A: devpath=0\n
|
|
||||||
L: driver=../../../../bus/usb/drivers/usb
|
|
||||||
L: firmware_node=../../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:4c/device:4d
|
|
||||||
A: idProduct=0002\n
|
|
||||||
A: idVendor=1d6b\n
|
|
||||||
A: interface_authorized_default=1\n
|
|
||||||
A: ltm_capable=no\n
|
|
||||||
A: manufacturer=Linux 5.13.12-200.fc34.x86_64 xhci-hcd\n
|
|
||||||
A: maxchild=16\n
|
|
||||||
A: power/active_duration=767293591\n
|
|
||||||
A: power/autosuspend=0\n
|
|
||||||
A: power/autosuspend_delay_ms=0\n
|
|
||||||
A: power/connected_duration=767293591\n
|
|
||||||
A: power/control=auto\n
|
|
||||||
A: power/level=auto\n
|
|
||||||
A: power/runtime_active_time=767293588\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=0\n
|
|
||||||
A: power/wakeup=disabled\n
|
|
||||||
A: power/wakeup_abort_count=\n
|
|
||||||
A: power/wakeup_active=\n
|
|
||||||
A: power/wakeup_active_count=\n
|
|
||||||
A: power/wakeup_count=\n
|
|
||||||
A: power/wakeup_expire_count=\n
|
|
||||||
A: power/wakeup_last_time_ms=\n
|
|
||||||
A: power/wakeup_max_time_ms=\n
|
|
||||||
A: power/wakeup_total_time_ms=\n
|
|
||||||
A: product=xHCI Host Controller\n
|
|
||||||
A: quirks=0x0\n
|
|
||||||
A: removable=unknown\n
|
|
||||||
A: rx_lanes=1\n
|
|
||||||
A: serial=0000:00:14.0\n
|
|
||||||
A: speed=480\n
|
|
||||||
A: tx_lanes=1\n
|
|
||||||
A: urbnum=1086\n
|
|
||||||
A: version= 2.00\n
|
|
||||||
|
|
||||||
P: /devices/pci0000:00/0000:00:14.0
|
|
||||||
E: DRIVER=xhci_hcd
|
|
||||||
E: PCI_CLASS=C0330
|
|
||||||
E: PCI_ID=8086:A36D
|
|
||||||
E: PCI_SUBSYS_ID=17AA:312A
|
|
||||||
E: PCI_SLOT_NAME=0000:00:14.0
|
|
||||||
E: MODALIAS=pci:v00008086d0000A36Dsv000017AAsd0000312Abc0Csc03i30
|
|
||||||
E: SUBSYSTEM=pci
|
|
||||||
E: ID_PCI_CLASS_FROM_DATABASE=Serial bus controller
|
|
||||||
E: ID_PCI_SUBCLASS_FROM_DATABASE=USB controller
|
|
||||||
E: ID_PCI_INTERFACE_FROM_DATABASE=XHCI
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Intel Corporation
|
|
||||||
E: ID_MODEL_FROM_DATABASE=Cannon Lake PCH USB 3.1 xHCI Host Controller
|
|
||||||
A: ari_enabled=0\n
|
|
||||||
A: broken_parity_status=0\n
|
|
||||||
A: class=0x0c0330\n
|
|
||||||
H: config=86806DA3060490021030030C00008000040032B1000000000000000000000000000000000000000000000000AA172A31000000007000000000000000FF010000FD0134808FC6FF8300000000000000007F6DDC0F00000000560BBD0700000000316000000000000000000000000000000180C2C1080000000000000000000000059087009802E0FE0000000000000000090014F01000400100000000C10A080000080E00001800008F400200000100006000000008020000010000000400000090000000001800000005000000000000000300000C0000004000000080000000030000000000000000000000000000000000000000000000B50F110112000000
|
|
||||||
A: consistent_dma_mask_bits=64\n
|
|
||||||
A: d3cold_allowed=1\n
|
|
||||||
A: dbc=disabled\n
|
|
||||||
A: device=0xa36d\n
|
|
||||||
A: dma_mask_bits=64\n
|
|
||||||
L: driver=../../../bus/pci/drivers/xhci_hcd
|
|
||||||
A: driver_override=(null)\n
|
|
||||||
A: enable=1\n
|
|
||||||
L: firmware_node=../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:4c
|
|
||||||
A: index=3\n
|
|
||||||
A: irq=125\n
|
|
||||||
A: label=Onboard - Other\n
|
|
||||||
A: local_cpulist=0-5\n
|
|
||||||
A: local_cpus=3f\n
|
|
||||||
A: modalias=pci:v00008086d0000A36Dsv000017AAsd0000312Abc0Csc03i30\n
|
|
||||||
A: msi_bus=1\n
|
|
||||||
A: msi_irqs/125=msi\n
|
|
||||||
A: numa_node=-1\n
|
|
||||||
A: pools=poolinfo - 0.1\nbuffer-2048 0 0 2048 0\nbuffer-512 0 0 512 0\nbuffer-128 0 0 128 0\nbuffer-32 0 0 32 0\nxHCI 1KB stream ctx arrays 0 0 1024 0\nxHCI 256 byte stream ctx arrays 0 0 256 0\nxHCI input/output contexts 15 20 2112 20\nxHCI ring segments 46 76 4096 76\nbuffer-2048 0 32 2048 16\nbuffer-512 0 32 512 4\nbuffer-128 3 32 128 1\nbuffer-32 0 128 32 1\n
|
|
||||||
A: power/control=on\n
|
|
||||||
A: power/runtime_active_time=767293736\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=0\n
|
|
||||||
A: power/wakeup=enabled\n
|
|
||||||
A: power/wakeup_abort_count=0\n
|
|
||||||
A: power/wakeup_active=0\n
|
|
||||||
A: power/wakeup_active_count=59\n
|
|
||||||
A: power/wakeup_count=0\n
|
|
||||||
A: power/wakeup_expire_count=59\n
|
|
||||||
A: power/wakeup_last_time_ms=763021754\n
|
|
||||||
A: power/wakeup_max_time_ms=108\n
|
|
||||||
A: power/wakeup_total_time_ms=6149\n
|
|
||||||
A: power_state=D0\n
|
|
||||||
A: resource=0x00000000b1320000 0x00000000b132ffff 0x0000000000140204\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n
|
|
||||||
A: revision=0x10\n
|
|
||||||
A: subsystem_device=0x312a\n
|
|
||||||
A: subsystem_vendor=0x17aa\n
|
|
||||||
A: vendor=0x8086\n
|
|
||||||
|
|
|
@ -1,162 +0,0 @@
|
||||||
#!/usr/bin/python3
|
|
||||||
|
|
||||||
BUILDDIR='@BUILDDIR@'
|
|
||||||
SRCDIR='@SRCDIR@'
|
|
||||||
|
|
||||||
import os
|
|
||||||
import sys
|
|
||||||
import signal
|
|
||||||
library_path = BUILDDIR + '/libfprint/'
|
|
||||||
|
|
||||||
# Relaunch ourselves with a changed environment so
|
|
||||||
# that we're loading the development version of libfprint
|
|
||||||
if 'LD_LIBRARY_PATH' not in os.environ or not library_path in os.environ['LD_LIBRARY_PATH']:
|
|
||||||
os.environ['LD_LIBRARY_PATH'] = library_path
|
|
||||||
os.environ['GI_TYPELIB_PATH'] = f'{BUILDDIR}/libfprint/'
|
|
||||||
os.environ['FP_DEVICE_EMULATION'] = '1'
|
|
||||||
try:
|
|
||||||
os.execv(sys.argv[0], sys.argv)
|
|
||||||
except Exception as e:
|
|
||||||
print('Could not run script with new library path')
|
|
||||||
sys.exit(1)
|
|
||||||
|
|
||||||
import gi
|
|
||||||
gi.require_version('FPrint', '2.0')
|
|
||||||
from gi.repository import FPrint
|
|
||||||
|
|
||||||
gi.require_version('GUsb', '1.0')
|
|
||||||
from gi.repository import GUsb
|
|
||||||
|
|
||||||
import re
|
|
||||||
import shutil
|
|
||||||
import subprocess
|
|
||||||
import tempfile
|
|
||||||
import time
|
|
||||||
|
|
||||||
def print_usage():
|
|
||||||
print(f'Usage: {sys.argv[0]} driver [test-variant-name]')
|
|
||||||
print('A test variant name is optional, and must be all lower case letters, or dashes, with no spaces')
|
|
||||||
print(f'The captured data will be stored in {SRCDIR}/tests/[driver name]-[test variant name]')
|
|
||||||
|
|
||||||
if len(sys.argv) > 3:
|
|
||||||
print_usage()
|
|
||||||
sys.exit(1)
|
|
||||||
|
|
||||||
driver_name = sys.argv[1]
|
|
||||||
os.environ['FP_DRIVERS_WHITELIST'] = driver_name
|
|
||||||
|
|
||||||
test_variant = None
|
|
||||||
if len(sys.argv) == 3:
|
|
||||||
valid_re = re.compile('[a-z-]*')
|
|
||||||
test_variant = sys.argv[2]
|
|
||||||
if (not valid_re.match(test_variant) or
|
|
||||||
test_variant.startswith('-') or
|
|
||||||
test_variant.endswith('-')):
|
|
||||||
print(f'Invalid variant name {test_variant}\n')
|
|
||||||
print_usage()
|
|
||||||
sys.exit(1)
|
|
||||||
|
|
||||||
# Check that running as root
|
|
||||||
|
|
||||||
if os.geteuid() != 0:
|
|
||||||
print(f'{sys.argv[0]} is expected to be run as root')
|
|
||||||
sys.exit(1)
|
|
||||||
|
|
||||||
# Check that tshark is available
|
|
||||||
|
|
||||||
tshark = shutil.which('tshark')
|
|
||||||
if not tshark:
|
|
||||||
print("The 'tshark' WireShark command-line tool must be installed to capture USB traffic")
|
|
||||||
sys.exit(1)
|
|
||||||
|
|
||||||
# Find the fingerprint reader
|
|
||||||
ctx = FPrint.Context()
|
|
||||||
ctx.enumerate()
|
|
||||||
devices = ctx.get_devices()
|
|
||||||
if len(devices) == 0:
|
|
||||||
print('Could not find a supported fingerprint reader')
|
|
||||||
sys.exit(1)
|
|
||||||
elif len(devices) > 1:
|
|
||||||
print('Capture requires a single supported fingerprint reader to be plugged in')
|
|
||||||
sys.exit(1)
|
|
||||||
|
|
||||||
test_name = driver_name
|
|
||||||
if test_variant:
|
|
||||||
test_name = driver_name + '-' + test_variant
|
|
||||||
usb_device = devices[0].get_property('fpi-usb-device')
|
|
||||||
bus_num = usb_device.get_bus()
|
|
||||||
device_num = usb_device.get_address()
|
|
||||||
|
|
||||||
print(f'### Detected USB device /dev/bus/usb/{bus_num:03d}/{device_num:03d}')
|
|
||||||
|
|
||||||
# Make directory
|
|
||||||
|
|
||||||
test_dir = SRCDIR + '/tests/' + test_name
|
|
||||||
os.makedirs(test_dir, mode=0o775, exist_ok=True)
|
|
||||||
|
|
||||||
# Capture device info
|
|
||||||
|
|
||||||
args = ['umockdev-record', f'/dev/bus/usb/{bus_num:03d}/{device_num:03d}']
|
|
||||||
device_out = open(test_dir + '/device', 'w')
|
|
||||||
process = subprocess.Popen(args, stdout=device_out)
|
|
||||||
process.wait()
|
|
||||||
|
|
||||||
# Run capture
|
|
||||||
# https://osqa-ask.wireshark.org/questions/53919/how-can-i-precisely-specify-a-usb-device-to-capture-with-tshark/
|
|
||||||
|
|
||||||
print(f'### Starting USB capture on usbmon{bus_num}')
|
|
||||||
capture_pid = os.fork()
|
|
||||||
assert(capture_pid >= 0)
|
|
||||||
|
|
||||||
unfiltered_cap_path = os.path.join(tempfile.gettempdir(), 'capture-unfiltered.pcapng')
|
|
||||||
if capture_pid == 0:
|
|
||||||
os.setpgrp()
|
|
||||||
args = ['tshark', '-q', '-i', f'usbmon{bus_num}', '-w', unfiltered_cap_path]
|
|
||||||
os.execv(tshark, args)
|
|
||||||
|
|
||||||
# Wait 1 sec to settle (we can assume setpgrp happened)
|
|
||||||
time.sleep(1)
|
|
||||||
|
|
||||||
print('### Capturing fingerprint, please swipe or press your finger on the reader')
|
|
||||||
with subprocess.Popen(['python3', SRCDIR + '/tests/capture.py', test_dir + '/capture.png']) as capture_process:
|
|
||||||
capture_process.wait()
|
|
||||||
if capture_process.returncode != 0:
|
|
||||||
print('Failed to capture fingerprint')
|
|
||||||
os.killpg(capture_pid, signal.SIGKILL)
|
|
||||||
sys.exit(1)
|
|
||||||
|
|
||||||
def t_waitpid(pid, timeout):
|
|
||||||
timeout = time.time() + timeout
|
|
||||||
r = os.waitpid(pid, os.WNOHANG)
|
|
||||||
while timeout > time.time() and r[0] == 0:
|
|
||||||
time.sleep(0.1)
|
|
||||||
r = os.waitpid(pid, os.WNOHANG)
|
|
||||||
|
|
||||||
return r
|
|
||||||
|
|
||||||
os.kill(capture_pid, signal.SIGTERM)
|
|
||||||
try:
|
|
||||||
r = t_waitpid(capture_pid, 2)
|
|
||||||
# Kill if nothing died
|
|
||||||
if r[0] == 0:
|
|
||||||
os.kill(capture_pid, signal.SIGKILL)
|
|
||||||
except ChildProcessError:
|
|
||||||
pass
|
|
||||||
|
|
||||||
try:
|
|
||||||
while True:
|
|
||||||
r = t_waitpid(-capture_pid, timeout=2)
|
|
||||||
# Kill the process group, if nothing died (and there are children)
|
|
||||||
if r[0] == 0:
|
|
||||||
os.killpg(capture_pid, signal.SIGKILL)
|
|
||||||
except ChildProcessError:
|
|
||||||
pass
|
|
||||||
|
|
||||||
# Filter the capture
|
|
||||||
print(f'\n### Saving USB capture as test case {test_name}')
|
|
||||||
args = ['tshark', '-r', unfiltered_cap_path, '-Y', f'usb.bus_id == {bus_num} and usb.device_address == {device_num}',
|
|
||||||
'-w', test_dir + '/capture.pcapng']
|
|
||||||
with subprocess.Popen(args, stderr=subprocess.DEVNULL) as filter_process:
|
|
||||||
filter_process.wait()
|
|
||||||
|
|
||||||
print(f"\nDone! Don't forget to add {test_name} to tests/meson.build")
|
|
Binary file not shown.
Binary file not shown.
Before Width: | Height: | Size: 100 KiB |
|
@ -1,228 +0,0 @@
|
||||||
P: /devices/pci0000:00/0000:00:14.0/usb1/1-9
|
|
||||||
N: bus/usb/001/005=12011001000000087A1C700541100102030109022000010100A0320904000002FF0000000705830240000007050402400003
|
|
||||||
E: DEVNAME=/dev/bus/usb/001/005
|
|
||||||
E: DEVTYPE=usb_device
|
|
||||||
E: DRIVER=usb
|
|
||||||
E: PRODUCT=1c7a/570/1041
|
|
||||||
E: TYPE=0/0/0
|
|
||||||
E: BUSNUM=001
|
|
||||||
E: DEVNUM=005
|
|
||||||
E: MAJOR=189
|
|
||||||
E: MINOR=4
|
|
||||||
E: SUBSYSTEM=usb
|
|
||||||
E: ID_VENDOR=EgisTec
|
|
||||||
E: ID_VENDOR_ENC=EgisTec
|
|
||||||
E: ID_VENDOR_ID=1c7a
|
|
||||||
E: ID_MODEL=EgisTec_Touch_Fingerprint_Sensor
|
|
||||||
E: ID_MODEL_ENC=EgisTec\x20Touch\x20Fingerprint\x20Sensor
|
|
||||||
E: ID_MODEL_ID=0570
|
|
||||||
E: ID_REVISION=1041
|
|
||||||
E: ID_SERIAL=EgisTec_EgisTec_Touch_Fingerprint_Sensor_W700B41B
|
|
||||||
E: ID_SERIAL_SHORT=W700B41B
|
|
||||||
E: ID_BUS=usb
|
|
||||||
E: ID_USB_INTERFACES=:ff0000:
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=LighTuning Technology Inc.
|
|
||||||
E: ID_PATH=pci-0000:00:14.0-usb-0:9
|
|
||||||
E: ID_PATH_TAG=pci-0000_00_14_0-usb-0_9
|
|
||||||
E: LIBFPRINT_DRIVER=Hardcoded whitelist
|
|
||||||
A: authorized=1
|
|
||||||
A: avoid_reset_quirk=0
|
|
||||||
A: bConfigurationValue=1
|
|
||||||
A: bDeviceClass=00
|
|
||||||
A: bDeviceProtocol=00
|
|
||||||
A: bDeviceSubClass=00
|
|
||||||
A: bMaxPacketSize0=8
|
|
||||||
A: bMaxPower=100mA
|
|
||||||
A: bNumConfigurations=1
|
|
||||||
A: bNumInterfaces= 1
|
|
||||||
A: bcdDevice=1041
|
|
||||||
A: bmAttributes=a0
|
|
||||||
A: busnum=1
|
|
||||||
A: configuration=
|
|
||||||
H: descriptors=12011001000000087A1C700541100102030109022000010100A0320904000002FF0000000705830240000007050402400003
|
|
||||||
A: dev=189:4
|
|
||||||
A: devnum=5
|
|
||||||
A: devpath=9
|
|
||||||
L: driver=../../../../../bus/usb/drivers/usb
|
|
||||||
A: idProduct=0570
|
|
||||||
A: idVendor=1c7a
|
|
||||||
A: ltm_capable=no
|
|
||||||
A: manufacturer=EgisTec
|
|
||||||
A: maxchild=0
|
|
||||||
L: port=../1-0:1.0/usb1-port9
|
|
||||||
A: power/active_duration=362352
|
|
||||||
A: power/async=enabled
|
|
||||||
A: power/autosuspend=2
|
|
||||||
A: power/autosuspend_delay_ms=2000
|
|
||||||
A: power/connected_duration=5526124
|
|
||||||
A: power/control=auto
|
|
||||||
A: power/level=auto
|
|
||||||
A: power/persist=1
|
|
||||||
A: power/runtime_active_kids=0
|
|
||||||
A: power/runtime_active_time=365097
|
|
||||||
A: power/runtime_enabled=enabled
|
|
||||||
A: power/runtime_status=active
|
|
||||||
A: power/runtime_suspended_time=5160752
|
|
||||||
A: power/runtime_usage=0
|
|
||||||
A: power/wakeup=disabled
|
|
||||||
A: power/wakeup_abort_count=
|
|
||||||
A: power/wakeup_active=
|
|
||||||
A: power/wakeup_active_count=
|
|
||||||
A: power/wakeup_count=
|
|
||||||
A: power/wakeup_expire_count=
|
|
||||||
A: power/wakeup_last_time_ms=
|
|
||||||
A: power/wakeup_max_time_ms=
|
|
||||||
A: power/wakeup_total_time_ms=
|
|
||||||
A: product=EgisTec Touch Fingerprint Sensor
|
|
||||||
A: quirks=0x0
|
|
||||||
A: removable=fixed
|
|
||||||
A: rx_lanes=1
|
|
||||||
A: serial=W700B41B
|
|
||||||
A: speed=12
|
|
||||||
A: tx_lanes=1
|
|
||||||
A: urbnum=8040
|
|
||||||
A: version= 1.10
|
|
||||||
|
|
||||||
P: /devices/pci0000:00/0000:00:14.0/usb1
|
|
||||||
N: bus/usb/001/001=12010002090001406B1D020008050302010109021900010100E0000904000001090000000705810304000C
|
|
||||||
E: DEVNAME=/dev/bus/usb/001/001
|
|
||||||
E: DEVTYPE=usb_device
|
|
||||||
E: DRIVER=usb
|
|
||||||
E: PRODUCT=1d6b/2/508
|
|
||||||
E: TYPE=9/0/1
|
|
||||||
E: BUSNUM=001
|
|
||||||
E: DEVNUM=001
|
|
||||||
E: MAJOR=189
|
|
||||||
E: MINOR=0
|
|
||||||
E: SUBSYSTEM=usb
|
|
||||||
E: ID_VENDOR=Linux_5.8.0-59-generic_xhci-hcd
|
|
||||||
E: ID_VENDOR_ENC=Linux\x205.8.0-59-generic\x20xhci-hcd
|
|
||||||
E: ID_VENDOR_ID=1d6b
|
|
||||||
E: ID_MODEL=xHCI_Host_Controller
|
|
||||||
E: ID_MODEL_ENC=xHCI\x20Host\x20Controller
|
|
||||||
E: ID_MODEL_ID=0002
|
|
||||||
E: ID_REVISION=0508
|
|
||||||
E: ID_SERIAL=Linux_5.8.0-59-generic_xhci-hcd_xHCI_Host_Controller_0000:00:14.0
|
|
||||||
E: ID_SERIAL_SHORT=0000:00:14.0
|
|
||||||
E: ID_BUS=usb
|
|
||||||
E: ID_USB_INTERFACES=:090000:
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Linux Foundation
|
|
||||||
E: ID_MODEL_FROM_DATABASE=2.0 root hub
|
|
||||||
E: ID_PATH=pci-0000:00:14.0
|
|
||||||
E: ID_PATH_TAG=pci-0000_00_14_0
|
|
||||||
E: ID_FOR_SEAT=usb-pci-0000_00_14_0
|
|
||||||
E: TAGS=:seat:
|
|
||||||
A: authorized=1
|
|
||||||
A: authorized_default=1
|
|
||||||
A: avoid_reset_quirk=0
|
|
||||||
A: bConfigurationValue=1
|
|
||||||
A: bDeviceClass=09
|
|
||||||
A: bDeviceProtocol=01
|
|
||||||
A: bDeviceSubClass=00
|
|
||||||
A: bMaxPacketSize0=64
|
|
||||||
A: bMaxPower=0mA
|
|
||||||
A: bNumConfigurations=1
|
|
||||||
A: bNumInterfaces= 1
|
|
||||||
A: bcdDevice=0508
|
|
||||||
A: bmAttributes=e0
|
|
||||||
A: busnum=1
|
|
||||||
A: configuration=
|
|
||||||
H: descriptors=12010002090001406B1D020008050302010109021900010100E0000904000001090000000705810304000C
|
|
||||||
A: dev=189:0
|
|
||||||
A: devnum=1
|
|
||||||
A: devpath=0
|
|
||||||
L: driver=../../../../bus/usb/drivers/usb
|
|
||||||
A: idProduct=0002
|
|
||||||
A: idVendor=1d6b
|
|
||||||
A: interface_authorized_default=1
|
|
||||||
A: ltm_capable=no
|
|
||||||
A: manufacturer=Linux 5.8.0-59-generic xhci-hcd
|
|
||||||
A: maxchild=12
|
|
||||||
A: power/active_duration=378024
|
|
||||||
A: power/async=enabled
|
|
||||||
A: power/autosuspend=0
|
|
||||||
A: power/autosuspend_delay_ms=0
|
|
||||||
A: power/connected_duration=5527220
|
|
||||||
A: power/control=auto
|
|
||||||
A: power/level=auto
|
|
||||||
A: power/runtime_active_kids=1
|
|
||||||
A: power/runtime_active_time=377962
|
|
||||||
A: power/runtime_enabled=enabled
|
|
||||||
A: power/runtime_status=active
|
|
||||||
A: power/runtime_suspended_time=5149253
|
|
||||||
A: power/runtime_usage=0
|
|
||||||
A: power/wakeup=disabled
|
|
||||||
A: power/wakeup_abort_count=
|
|
||||||
A: power/wakeup_active=
|
|
||||||
A: power/wakeup_active_count=
|
|
||||||
A: power/wakeup_count=
|
|
||||||
A: power/wakeup_expire_count=
|
|
||||||
A: power/wakeup_last_time_ms=
|
|
||||||
A: power/wakeup_max_time_ms=
|
|
||||||
A: power/wakeup_total_time_ms=
|
|
||||||
A: product=xHCI Host Controller
|
|
||||||
A: quirks=0x0
|
|
||||||
A: removable=unknown
|
|
||||||
A: rx_lanes=1
|
|
||||||
A: serial=0000:00:14.0
|
|
||||||
A: speed=480
|
|
||||||
A: tx_lanes=1
|
|
||||||
A: urbnum=956
|
|
||||||
A: version= 2.00
|
|
||||||
|
|
||||||
P: /devices/pci0000:00/0000:00:14.0
|
|
||||||
E: DRIVER=xhci_hcd
|
|
||||||
E: PCI_CLASS=C0330
|
|
||||||
E: PCI_ID=8086:9D2F
|
|
||||||
E: PCI_SUBSYS_ID=1025:118E
|
|
||||||
E: PCI_SLOT_NAME=0000:00:14.0
|
|
||||||
E: MODALIAS=pci:v00008086d00009D2Fsv00001025sd0000118Ebc0Csc03i30
|
|
||||||
E: SUBSYSTEM=pci
|
|
||||||
E: ID_PCI_CLASS_FROM_DATABASE=Serial bus controller
|
|
||||||
E: ID_PCI_SUBCLASS_FROM_DATABASE=USB controller
|
|
||||||
E: ID_PCI_INTERFACE_FROM_DATABASE=XHCI
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Intel Corporation
|
|
||||||
E: ID_MODEL_FROM_DATABASE=Sunrise Point-LP USB 3.0 xHCI Controller
|
|
||||||
A: ari_enabled=0
|
|
||||||
A: broken_parity_status=0
|
|
||||||
A: class=0x0c0330
|
|
||||||
H: config=86802F9D060490022130030C00008000040021A400000000000000000000000000000000000000000000000025108E11000000007000000000000000FF010000
|
|
||||||
A: consistent_dma_mask_bits=64
|
|
||||||
A: d3cold_allowed=1
|
|
||||||
A: dbc=disabled
|
|
||||||
A: device=0x9d2f
|
|
||||||
A: dma_mask_bits=64
|
|
||||||
L: driver=../../../bus/pci/drivers/xhci_hcd
|
|
||||||
A: driver_override=(null)
|
|
||||||
A: enable=1
|
|
||||||
A: irq=127
|
|
||||||
A: local_cpulist=0-7
|
|
||||||
A: local_cpus=ff
|
|
||||||
A: modalias=pci:v00008086d00009D2Fsv00001025sd0000118Ebc0Csc03i30
|
|
||||||
A: msi_bus=1
|
|
||||||
A: msi_irqs/127=msi
|
|
||||||
A: numa_node=-1
|
|
||||||
A: pools=poolinfo - 0.1\nbuffer-2048 0 0 2048 0\nbuffer-512 0 0 512 0\nbuffer-128 0 0 128 0\nbuffer-32 0 0 32 0\nxHCI 1KB stream ctx arrays 0 0 1024 0\nxHCI 256 byte stream ctx arrays 0 0 256 0\nxHCI input/output contexts 9 10 2112 10\nxHCI ring segments 32 36 4096 36\nbuffer-2048 1 2 2048 1\nbuffer-512 0 0 512 0\nbuffer-128 0 0 128 0\nbuffer-32 0 0 32 0
|
|
||||||
A: power/async=enabled
|
|
||||||
A: power/control=on
|
|
||||||
A: power/runtime_active_kids=1
|
|
||||||
A: power/runtime_active_time=5524703
|
|
||||||
A: power/runtime_enabled=forbidden
|
|
||||||
A: power/runtime_status=active
|
|
||||||
A: power/runtime_suspended_time=3373
|
|
||||||
A: power/runtime_usage=1
|
|
||||||
A: power/wakeup=enabled
|
|
||||||
A: power/wakeup_abort_count=0
|
|
||||||
A: power/wakeup_active=0
|
|
||||||
A: power/wakeup_active_count=0
|
|
||||||
A: power/wakeup_count=0
|
|
||||||
A: power/wakeup_expire_count=0
|
|
||||||
A: power/wakeup_last_time_ms=0
|
|
||||||
A: power/wakeup_max_time_ms=0
|
|
||||||
A: power/wakeup_total_time_ms=0
|
|
||||||
A: resource=0x00000000a4210000 0x00000000a421ffff 0x0000000000140204\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000
|
|
||||||
A: revision=0x21
|
|
||||||
A: subsystem_device=0x118e
|
|
||||||
A: subsystem_vendor=0x1025
|
|
||||||
A: vendor=0x8086
|
|
||||||
|
|
Binary file not shown.
Binary file not shown.
Before Width: | Height: | Size: 48 KiB |
|
@ -1,217 +0,0 @@
|
||||||
P: /devices/pci0000:00/0000:00:14.0/usb1/1-10
|
|
||||||
N: bus/usb/001/045=1201000200000008F304260C40010102000109023E0001010080320904000005FF0000000921100100012215000705810240000107050102400001070582024000010705830240000107050302400001
|
|
||||||
E: DEVNAME=/dev/bus/usb/001/045
|
|
||||||
E: DEVTYPE=usb_device
|
|
||||||
E: DRIVER=usb
|
|
||||||
E: PRODUCT=4f3/c26/140
|
|
||||||
E: TYPE=0/0/0
|
|
||||||
E: BUSNUM=001
|
|
||||||
E: DEVNUM=045
|
|
||||||
E: MAJOR=189
|
|
||||||
E: MINOR=44
|
|
||||||
E: SUBSYSTEM=usb
|
|
||||||
E: ID_VENDOR=ELAN
|
|
||||||
E: ID_VENDOR_ENC=ELAN
|
|
||||||
E: ID_VENDOR_ID=04f3
|
|
||||||
E: ID_MODEL=ELAN:Fingerprint
|
|
||||||
E: ID_MODEL_ENC=ELAN:Fingerprint
|
|
||||||
E: ID_MODEL_ID=0c26
|
|
||||||
E: ID_REVISION=0140
|
|
||||||
E: ID_SERIAL=ELAN_ELAN:Fingerprint
|
|
||||||
E: ID_BUS=usb
|
|
||||||
E: ID_USB_INTERFACES=:ff0000:
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Elan Microelectronics Corp.
|
|
||||||
E: ID_AUTOSUSPEND=1
|
|
||||||
E: ID_PATH=pci-0000:00:14.0-usb-0:10
|
|
||||||
E: ID_PATH_TAG=pci-0000_00_14_0-usb-0_10
|
|
||||||
E: LIBFPRINT_DRIVER=ElanTech Fingerprint Sensor
|
|
||||||
E: ID_FOR_SEAT=usb-pci-0000_00_14_0-usb-0_10
|
|
||||||
E: TAGS=:seat:
|
|
||||||
E: CURRENT_TAGS=:seat:
|
|
||||||
A: authorized=1\n
|
|
||||||
A: avoid_reset_quirk=0\n
|
|
||||||
A: bConfigurationValue=1\n
|
|
||||||
A: bDeviceClass=00\n
|
|
||||||
A: bDeviceProtocol=00\n
|
|
||||||
A: bDeviceSubClass=00\n
|
|
||||||
A: bMaxPacketSize0=8\n
|
|
||||||
A: bMaxPower=100mA\n
|
|
||||||
A: bNumConfigurations=1\n
|
|
||||||
A: bNumInterfaces= 1\n
|
|
||||||
A: bcdDevice=0140\n
|
|
||||||
A: bmAttributes=80\n
|
|
||||||
A: busnum=1\n
|
|
||||||
A: configuration=
|
|
||||||
H: descriptors=1201000200000008F304260C40010102000109023E0001010080320904000005FF0000000921100100012215000705810240000107050102400001070582024000010705830240000107050302400001
|
|
||||||
A: dev=189:44\n
|
|
||||||
A: devnum=45\n
|
|
||||||
A: devpath=10\n
|
|
||||||
L: driver=../../../../../bus/usb/drivers/usb
|
|
||||||
L: firmware_node=../../../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:4c/device:4d/device:57
|
|
||||||
A: idProduct=0c26\n
|
|
||||||
A: idVendor=04f3\n
|
|
||||||
A: ltm_capable=no\n
|
|
||||||
A: manufacturer=ELAN\n
|
|
||||||
A: maxchild=0\n
|
|
||||||
L: port=../1-0:1.0/usb1-port10
|
|
||||||
A: power/active_duration=21526\n
|
|
||||||
A: power/autosuspend=2\n
|
|
||||||
A: power/autosuspend_delay_ms=2000\n
|
|
||||||
A: power/connected_duration=96442\n
|
|
||||||
A: power/control=auto\n
|
|
||||||
A: power/level=auto\n
|
|
||||||
A: power/persist=0\n
|
|
||||||
A: power/runtime_active_time=21572\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=74628\n
|
|
||||||
A: product=ELAN:Fingerprint\n
|
|
||||||
A: quirks=0x0\n
|
|
||||||
A: removable=removable\n
|
|
||||||
A: rx_lanes=1\n
|
|
||||||
A: speed=12\n
|
|
||||||
A: tx_lanes=1\n
|
|
||||||
A: urbnum=103\n
|
|
||||||
A: version= 2.00\n
|
|
||||||
|
|
||||||
P: /devices/pci0000:00/0000:00:14.0/usb1
|
|
||||||
N: bus/usb/001/001=12010002090001406B1D020013050302010109021900010100E0000904000001090000000705810304000C
|
|
||||||
E: DEVNAME=/dev/bus/usb/001/001
|
|
||||||
E: DEVTYPE=usb_device
|
|
||||||
E: DRIVER=usb
|
|
||||||
E: PRODUCT=1d6b/2/513
|
|
||||||
E: TYPE=9/0/1
|
|
||||||
E: BUSNUM=001
|
|
||||||
E: DEVNUM=001
|
|
||||||
E: MAJOR=189
|
|
||||||
E: MINOR=0
|
|
||||||
E: SUBSYSTEM=usb
|
|
||||||
E: ID_VENDOR=Linux_5.13.12-200.fc34.x86_64_xhci-hcd
|
|
||||||
E: ID_VENDOR_ENC=Linux\x205.13.12-200.fc34.x86_64\x20xhci-hcd
|
|
||||||
E: ID_VENDOR_ID=1d6b
|
|
||||||
E: ID_MODEL=xHCI_Host_Controller
|
|
||||||
E: ID_MODEL_ENC=xHCI\x20Host\x20Controller
|
|
||||||
E: ID_MODEL_ID=0002
|
|
||||||
E: ID_REVISION=0513
|
|
||||||
E: ID_SERIAL=Linux_5.13.12-200.fc34.x86_64_xhci-hcd_xHCI_Host_Controller_0000:00:14.0
|
|
||||||
E: ID_SERIAL_SHORT=0000:00:14.0
|
|
||||||
E: ID_BUS=usb
|
|
||||||
E: ID_USB_INTERFACES=:090000:
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Linux Foundation
|
|
||||||
E: ID_AUTOSUSPEND=1
|
|
||||||
E: ID_MODEL_FROM_DATABASE=2.0 root hub
|
|
||||||
E: ID_PATH=pci-0000:00:14.0
|
|
||||||
E: ID_PATH_TAG=pci-0000_00_14_0
|
|
||||||
E: ID_FOR_SEAT=usb-pci-0000_00_14_0
|
|
||||||
E: TAGS=:seat:
|
|
||||||
E: CURRENT_TAGS=:seat:
|
|
||||||
A: authorized=1\n
|
|
||||||
A: authorized_default=1\n
|
|
||||||
A: avoid_reset_quirk=0\n
|
|
||||||
A: bConfigurationValue=1\n
|
|
||||||
A: bDeviceClass=09\n
|
|
||||||
A: bDeviceProtocol=01\n
|
|
||||||
A: bDeviceSubClass=00\n
|
|
||||||
A: bMaxPacketSize0=64\n
|
|
||||||
A: bMaxPower=0mA\n
|
|
||||||
A: bNumConfigurations=1\n
|
|
||||||
A: bNumInterfaces= 1\n
|
|
||||||
A: bcdDevice=0513\n
|
|
||||||
A: bmAttributes=e0\n
|
|
||||||
A: busnum=1\n
|
|
||||||
A: configuration=
|
|
||||||
H: descriptors=12010002090001406B1D020013050302010109021900010100E0000904000001090000000705810304000C
|
|
||||||
A: dev=189:0\n
|
|
||||||
A: devnum=1\n
|
|
||||||
A: devpath=0\n
|
|
||||||
L: driver=../../../../bus/usb/drivers/usb
|
|
||||||
L: firmware_node=../../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:4c/device:4d
|
|
||||||
A: idProduct=0002\n
|
|
||||||
A: idVendor=1d6b\n
|
|
||||||
A: interface_authorized_default=1\n
|
|
||||||
A: ltm_capable=no\n
|
|
||||||
A: manufacturer=Linux 5.13.12-200.fc34.x86_64 xhci-hcd\n
|
|
||||||
A: maxchild=16\n
|
|
||||||
A: power/active_duration=767973436\n
|
|
||||||
A: power/autosuspend=0\n
|
|
||||||
A: power/autosuspend_delay_ms=0\n
|
|
||||||
A: power/connected_duration=767973436\n
|
|
||||||
A: power/control=auto\n
|
|
||||||
A: power/level=auto\n
|
|
||||||
A: power/runtime_active_time=767973433\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=0\n
|
|
||||||
A: power/wakeup=disabled\n
|
|
||||||
A: power/wakeup_abort_count=\n
|
|
||||||
A: power/wakeup_active=\n
|
|
||||||
A: power/wakeup_active_count=\n
|
|
||||||
A: power/wakeup_count=\n
|
|
||||||
A: power/wakeup_expire_count=\n
|
|
||||||
A: power/wakeup_last_time_ms=\n
|
|
||||||
A: power/wakeup_max_time_ms=\n
|
|
||||||
A: power/wakeup_total_time_ms=\n
|
|
||||||
A: product=xHCI Host Controller\n
|
|
||||||
A: quirks=0x0\n
|
|
||||||
A: removable=unknown\n
|
|
||||||
A: rx_lanes=1\n
|
|
||||||
A: serial=0000:00:14.0\n
|
|
||||||
A: speed=480\n
|
|
||||||
A: tx_lanes=1\n
|
|
||||||
A: urbnum=1174\n
|
|
||||||
A: version= 2.00\n
|
|
||||||
|
|
||||||
P: /devices/pci0000:00/0000:00:14.0
|
|
||||||
E: DRIVER=xhci_hcd
|
|
||||||
E: PCI_CLASS=C0330
|
|
||||||
E: PCI_ID=8086:A36D
|
|
||||||
E: PCI_SUBSYS_ID=17AA:312A
|
|
||||||
E: PCI_SLOT_NAME=0000:00:14.0
|
|
||||||
E: MODALIAS=pci:v00008086d0000A36Dsv000017AAsd0000312Abc0Csc03i30
|
|
||||||
E: SUBSYSTEM=pci
|
|
||||||
E: ID_PCI_CLASS_FROM_DATABASE=Serial bus controller
|
|
||||||
E: ID_PCI_SUBCLASS_FROM_DATABASE=USB controller
|
|
||||||
E: ID_PCI_INTERFACE_FROM_DATABASE=XHCI
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Intel Corporation
|
|
||||||
E: ID_MODEL_FROM_DATABASE=Cannon Lake PCH USB 3.1 xHCI Host Controller
|
|
||||||
A: ari_enabled=0\n
|
|
||||||
A: broken_parity_status=0\n
|
|
||||||
A: class=0x0c0330\n
|
|
||||||
H: config=86806DA3060490021030030C00008000040032B1000000000000000000000000000000000000000000000000AA172A31000000007000000000000000FF010000FD0134808FC6FF8300000000000000007F6DDC0F00000000420DD90700000000316000000000000000000000000000000180C2C1080000000000000000000000059087009802E0FE0000000000000000090014F01000400100000000C10A080000080E00001800008F400200000100006000000008020000010000000400000090000000001800000005000000000000000300000C0000004000000080000000030000000000000000000000000000000000000000000000B50F110112000000
|
|
||||||
A: consistent_dma_mask_bits=64\n
|
|
||||||
A: d3cold_allowed=1\n
|
|
||||||
A: dbc=disabled\n
|
|
||||||
A: device=0xa36d\n
|
|
||||||
A: dma_mask_bits=64\n
|
|
||||||
L: driver=../../../bus/pci/drivers/xhci_hcd
|
|
||||||
A: driver_override=(null)\n
|
|
||||||
A: enable=1\n
|
|
||||||
L: firmware_node=../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:4c
|
|
||||||
A: index=3\n
|
|
||||||
A: irq=125\n
|
|
||||||
A: label=Onboard - Other\n
|
|
||||||
A: local_cpulist=0-5\n
|
|
||||||
A: local_cpus=3f\n
|
|
||||||
A: modalias=pci:v00008086d0000A36Dsv000017AAsd0000312Abc0Csc03i30\n
|
|
||||||
A: msi_bus=1\n
|
|
||||||
A: msi_irqs/125=msi\n
|
|
||||||
A: numa_node=-1\n
|
|
||||||
A: pools=poolinfo - 0.1\nbuffer-2048 0 0 2048 0\nbuffer-512 0 0 512 0\nbuffer-128 0 0 128 0\nbuffer-32 0 0 32 0\nxHCI 1KB stream ctx arrays 0 0 1024 0\nxHCI 256 byte stream ctx arrays 0 0 256 0\nxHCI input/output contexts 15 20 2112 20\nxHCI ring segments 52 76 4096 76\nbuffer-2048 0 32 2048 16\nbuffer-512 0 32 512 4\nbuffer-128 3 32 128 1\nbuffer-32 0 128 32 1\n
|
|
||||||
A: power/control=on\n
|
|
||||||
A: power/runtime_active_time=767973582\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=0\n
|
|
||||||
A: power/wakeup=enabled\n
|
|
||||||
A: power/wakeup_abort_count=0\n
|
|
||||||
A: power/wakeup_active=0\n
|
|
||||||
A: power/wakeup_active_count=59\n
|
|
||||||
A: power/wakeup_count=0\n
|
|
||||||
A: power/wakeup_expire_count=59\n
|
|
||||||
A: power/wakeup_last_time_ms=763021754\n
|
|
||||||
A: power/wakeup_max_time_ms=108\n
|
|
||||||
A: power/wakeup_total_time_ms=6149\n
|
|
||||||
A: power_state=D0\n
|
|
||||||
A: resource=0x00000000b1320000 0x00000000b132ffff 0x0000000000140204\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n
|
|
||||||
A: revision=0x10\n
|
|
||||||
A: subsystem_device=0x312a\n
|
|
||||||
A: subsystem_vendor=0x17aa\n
|
|
||||||
A: vendor=0x8086\n
|
|
||||||
|
|
Binary file not shown.
|
@ -1,83 +0,0 @@
|
||||||
#!/usr/bin/python3
|
|
||||||
|
|
||||||
import gi
|
|
||||||
gi.require_version('FPrint', '2.0')
|
|
||||||
from gi.repository import FPrint, GLib
|
|
||||||
|
|
||||||
ctx = GLib.main_context_default()
|
|
||||||
|
|
||||||
c = FPrint.Context()
|
|
||||||
c.enumerate()
|
|
||||||
devices = c.get_devices()
|
|
||||||
|
|
||||||
d = devices[0]
|
|
||||||
del devices
|
|
||||||
|
|
||||||
assert d.get_driver() == "elanmoc"
|
|
||||||
assert not d.has_feature(FPrint.DeviceFeature.CAPTURE)
|
|
||||||
assert d.has_feature(FPrint.DeviceFeature.IDENTIFY)
|
|
||||||
assert d.has_feature(FPrint.DeviceFeature.VERIFY)
|
|
||||||
assert not d.has_feature(FPrint.DeviceFeature.DUPLICATES_CHECK)
|
|
||||||
assert d.has_feature(FPrint.DeviceFeature.STORAGE)
|
|
||||||
assert d.has_feature(FPrint.DeviceFeature.STORAGE_LIST)
|
|
||||||
assert d.has_feature(FPrint.DeviceFeature.STORAGE_DELETE)
|
|
||||||
assert not d.has_feature(FPrint.DeviceFeature.STORAGE_CLEAR)
|
|
||||||
|
|
||||||
d.open_sync()
|
|
||||||
|
|
||||||
template = FPrint.Print.new(d)
|
|
||||||
|
|
||||||
def enroll_progress(*args):
|
|
||||||
#assert d.get_finger_status() == FPrint.FingerStatusFlags.NEEDED
|
|
||||||
print("finger status: ", d.get_finger_status())
|
|
||||||
print('enroll progress: ' + str(args))
|
|
||||||
|
|
||||||
def identify_done(dev, res):
|
|
||||||
global identified
|
|
||||||
identified = True
|
|
||||||
identify_match, identify_print = dev.identify_finish(res)
|
|
||||||
print('indentification_done: ', identify_match, identify_print)
|
|
||||||
assert identify_match.equal(identify_print)
|
|
||||||
|
|
||||||
# List, enroll, list, verify, identify, delete
|
|
||||||
print("enrolling")
|
|
||||||
assert d.get_finger_status() == FPrint.FingerStatusFlags.NONE
|
|
||||||
p = d.enroll_sync(template, None, enroll_progress, None)
|
|
||||||
assert d.get_finger_status() == FPrint.FingerStatusFlags.NONE
|
|
||||||
print("enroll done")
|
|
||||||
|
|
||||||
print("listing")
|
|
||||||
stored = d.list_prints_sync()
|
|
||||||
print("listing done")
|
|
||||||
assert len(stored) == 1
|
|
||||||
assert stored[0].equal(p)
|
|
||||||
print("verifying")
|
|
||||||
assert d.get_finger_status() == FPrint.FingerStatusFlags.NONE
|
|
||||||
verify_res, verify_print = d.verify_sync(p)
|
|
||||||
assert d.get_finger_status() == FPrint.FingerStatusFlags.NONE
|
|
||||||
print("verify done")
|
|
||||||
del p
|
|
||||||
assert verify_res == True
|
|
||||||
|
|
||||||
identified = False
|
|
||||||
deserialized_prints = []
|
|
||||||
for p in stored:
|
|
||||||
deserialized_prints.append(FPrint.Print.deserialize(p.serialize()))
|
|
||||||
assert deserialized_prints[-1].equal(p)
|
|
||||||
del stored
|
|
||||||
|
|
||||||
print('async identifying')
|
|
||||||
d.identify(deserialized_prints, callback=identify_done)
|
|
||||||
del deserialized_prints
|
|
||||||
|
|
||||||
while not identified:
|
|
||||||
ctx.iteration(True)
|
|
||||||
|
|
||||||
print("deleting")
|
|
||||||
d.delete_print_sync(p)
|
|
||||||
print("delete done")
|
|
||||||
|
|
||||||
d.close_sync()
|
|
||||||
|
|
||||||
del d
|
|
||||||
del c
|
|
|
@ -1,225 +0,0 @@
|
||||||
P: /devices/pci0000:00/0000:00:14.0/usb1/1-1
|
|
||||||
N: bus/usb/001/010=1201000200000040F3047E0C05030102000109025300010103A0320904000008FF0000000921100100012215000705810240000107050102400001070582024000010705020240000107058302400001070503024000010705840240000107050402400001
|
|
||||||
E: DEVNAME=/dev/bus/usb/001/010
|
|
||||||
E: DEVTYPE=usb_device
|
|
||||||
E: DRIVER=usb
|
|
||||||
E: PRODUCT=4f3/c7e/305
|
|
||||||
E: TYPE=0/0/0
|
|
||||||
E: BUSNUM=001
|
|
||||||
E: DEVNUM=010
|
|
||||||
E: MAJOR=189
|
|
||||||
E: MINOR=9
|
|
||||||
E: SUBSYSTEM=usb
|
|
||||||
E: ID_VENDOR=ELAN
|
|
||||||
E: ID_VENDOR_ENC=ELAN
|
|
||||||
E: ID_VENDOR_ID=04f3
|
|
||||||
E: ID_MODEL=ELAN:ARM-M4
|
|
||||||
E: ID_MODEL_ENC=ELAN:ARM-M4
|
|
||||||
E: ID_MODEL_ID=0c7e
|
|
||||||
E: ID_REVISION=0305
|
|
||||||
E: ID_SERIAL=ELAN_ELAN:ARM-M4
|
|
||||||
E: ID_BUS=usb
|
|
||||||
E: ID_USB_INTERFACES=:ff0000:
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Elan Microelectronics Corp.
|
|
||||||
E: ID_PATH=pci-0000:00:14.0-usb-0:1
|
|
||||||
E: ID_PATH_TAG=pci-0000_00_14_0-usb-0_1
|
|
||||||
A: authorized=1
|
|
||||||
A: avoid_reset_quirk=0
|
|
||||||
A: bConfigurationValue=1
|
|
||||||
A: bDeviceClass=00
|
|
||||||
A: bDeviceProtocol=00
|
|
||||||
A: bDeviceSubClass=00
|
|
||||||
A: bMaxPacketSize0=64
|
|
||||||
A: bMaxPower=100mA
|
|
||||||
A: bNumConfigurations=1
|
|
||||||
A: bNumInterfaces= 1
|
|
||||||
A: bcdDevice=0305
|
|
||||||
A: bmAttributes=a0
|
|
||||||
A: busnum=1
|
|
||||||
A: configuration=add909c9-e67e-4126-a6f7-1e31179e27d9
|
|
||||||
H: descriptors=1201000200000040F3047E0C05030102000109025300010103A0320904000008FF0000000921100100012215000705810240000107050102400001070582024000010705020240000107058302400001070503024000010705840240000107050402400001
|
|
||||||
A: dev=189:9
|
|
||||||
A: devnum=10
|
|
||||||
A: devpath=1
|
|
||||||
L: driver=../../../../../bus/usb/drivers/usb
|
|
||||||
A: idProduct=0c7e
|
|
||||||
A: idVendor=04f3
|
|
||||||
A: ltm_capable=no
|
|
||||||
A: manufacturer=ELAN
|
|
||||||
A: maxchild=0
|
|
||||||
L: port=../1-0:1.0/usb1-port1
|
|
||||||
A: power/active_duration=94712
|
|
||||||
A: power/async=enabled
|
|
||||||
A: power/autosuspend=2
|
|
||||||
A: power/autosuspend_delay_ms=2000
|
|
||||||
A: power/connected_duration=94712
|
|
||||||
A: power/control=on
|
|
||||||
A: power/level=on
|
|
||||||
A: power/persist=1
|
|
||||||
A: power/runtime_active_kids=0
|
|
||||||
A: power/runtime_active_time=94436
|
|
||||||
A: power/runtime_enabled=forbidden
|
|
||||||
A: power/runtime_status=active
|
|
||||||
A: power/runtime_suspended_time=0
|
|
||||||
A: power/runtime_usage=1
|
|
||||||
A: power/wakeup=disabled
|
|
||||||
A: power/wakeup_abort_count=
|
|
||||||
A: power/wakeup_active=
|
|
||||||
A: power/wakeup_active_count=
|
|
||||||
A: power/wakeup_count=
|
|
||||||
A: power/wakeup_expire_count=
|
|
||||||
A: power/wakeup_last_time_ms=
|
|
||||||
A: power/wakeup_max_time_ms=
|
|
||||||
A: power/wakeup_total_time_ms=
|
|
||||||
A: product=ELAN:ARM-M4
|
|
||||||
A: quirks=0x0
|
|
||||||
A: removable=removable
|
|
||||||
A: rx_lanes=1
|
|
||||||
A: speed=12
|
|
||||||
A: tx_lanes=1
|
|
||||||
A: urbnum=12
|
|
||||||
A: version= 2.00
|
|
||||||
|
|
||||||
P: /devices/pci0000:00/0000:00:14.0/usb1
|
|
||||||
N: bus/usb/001/001=12010002090001406B1D020004050302010109021900010100E0000904000001090000000705810304000C
|
|
||||||
E: DEVNAME=/dev/bus/usb/001/001
|
|
||||||
E: DEVTYPE=usb_device
|
|
||||||
E: DRIVER=usb
|
|
||||||
E: PRODUCT=1d6b/2/504
|
|
||||||
E: TYPE=9/0/1
|
|
||||||
E: BUSNUM=001
|
|
||||||
E: DEVNUM=001
|
|
||||||
E: MAJOR=189
|
|
||||||
E: MINOR=0
|
|
||||||
E: SUBSYSTEM=usb
|
|
||||||
E: ID_VENDOR=Linux_5.4.0-42-generic_xhci-hcd
|
|
||||||
E: ID_VENDOR_ENC=Linux\x205.4.0-42-generic\x20xhci-hcd
|
|
||||||
E: ID_VENDOR_ID=1d6b
|
|
||||||
E: ID_MODEL=xHCI_Host_Controller
|
|
||||||
E: ID_MODEL_ENC=xHCI\x20Host\x20Controller
|
|
||||||
E: ID_MODEL_ID=0002
|
|
||||||
E: ID_REVISION=0504
|
|
||||||
E: ID_SERIAL=Linux_5.4.0-42-generic_xhci-hcd_xHCI_Host_Controller_0000:00:14.0
|
|
||||||
E: ID_SERIAL_SHORT=0000:00:14.0
|
|
||||||
E: ID_BUS=usb
|
|
||||||
E: ID_USB_INTERFACES=:090000:
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Linux Foundation
|
|
||||||
E: ID_MODEL_FROM_DATABASE=2.0 root hub
|
|
||||||
E: ID_PATH=pci-0000:00:14.0
|
|
||||||
E: ID_PATH_TAG=pci-0000_00_14_0
|
|
||||||
E: ID_FOR_SEAT=usb-pci-0000_00_14_0
|
|
||||||
E: TAGS=:seat:
|
|
||||||
A: authorized=1
|
|
||||||
A: authorized_default=1
|
|
||||||
A: avoid_reset_quirk=0
|
|
||||||
A: bConfigurationValue=1
|
|
||||||
A: bDeviceClass=09
|
|
||||||
A: bDeviceProtocol=01
|
|
||||||
A: bDeviceSubClass=00
|
|
||||||
A: bMaxPacketSize0=64
|
|
||||||
A: bMaxPower=0mA
|
|
||||||
A: bNumConfigurations=1
|
|
||||||
A: bNumInterfaces= 1
|
|
||||||
A: bcdDevice=0504
|
|
||||||
A: bmAttributes=e0
|
|
||||||
A: busnum=1
|
|
||||||
A: configuration=
|
|
||||||
H: descriptors=12010002090001406B1D020004050302010109021900010100E0000904000001090000000705810304000C
|
|
||||||
A: dev=189:0
|
|
||||||
A: devnum=1
|
|
||||||
A: devpath=0
|
|
||||||
L: driver=../../../../bus/usb/drivers/usb
|
|
||||||
A: idProduct=0002
|
|
||||||
A: idVendor=1d6b
|
|
||||||
A: interface_authorized_default=1
|
|
||||||
A: ltm_capable=no
|
|
||||||
A: manufacturer=Linux 5.4.0-42-generic xhci-hcd
|
|
||||||
A: maxchild=12
|
|
||||||
A: power/active_duration=74604360
|
|
||||||
A: power/async=enabled
|
|
||||||
A: power/autosuspend=0
|
|
||||||
A: power/autosuspend_delay_ms=0
|
|
||||||
A: power/connected_duration=74606456
|
|
||||||
A: power/control=auto
|
|
||||||
A: power/level=auto
|
|
||||||
A: power/runtime_active_kids=4
|
|
||||||
A: power/runtime_active_time=74605838
|
|
||||||
A: power/runtime_enabled=enabled
|
|
||||||
A: power/runtime_status=active
|
|
||||||
A: power/runtime_suspended_time=0
|
|
||||||
A: power/runtime_usage=0
|
|
||||||
A: power/wakeup=disabled
|
|
||||||
A: power/wakeup_abort_count=
|
|
||||||
A: power/wakeup_active=
|
|
||||||
A: power/wakeup_active_count=
|
|
||||||
A: power/wakeup_count=
|
|
||||||
A: power/wakeup_expire_count=
|
|
||||||
A: power/wakeup_last_time_ms=
|
|
||||||
A: power/wakeup_max_time_ms=
|
|
||||||
A: power/wakeup_total_time_ms=
|
|
||||||
A: product=xHCI Host Controller
|
|
||||||
A: quirks=0x0
|
|
||||||
A: removable=unknown
|
|
||||||
A: rx_lanes=1
|
|
||||||
A: serial=0000:00:14.0
|
|
||||||
A: speed=480
|
|
||||||
A: tx_lanes=1
|
|
||||||
A: urbnum=490
|
|
||||||
A: version= 2.00
|
|
||||||
|
|
||||||
P: /devices/pci0000:00/0000:00:14.0
|
|
||||||
E: DRIVER=xhci_hcd
|
|
||||||
E: PCI_CLASS=C0330
|
|
||||||
E: PCI_ID=8086:9DED
|
|
||||||
E: PCI_SUBSYS_ID=103C:85EF
|
|
||||||
E: PCI_SLOT_NAME=0000:00:14.0
|
|
||||||
E: MODALIAS=pci:v00008086d00009DEDsv0000103Csd000085EFbc0Csc03i30
|
|
||||||
E: SUBSYSTEM=pci
|
|
||||||
E: ID_PCI_CLASS_FROM_DATABASE=Serial bus controller
|
|
||||||
E: ID_PCI_SUBCLASS_FROM_DATABASE=USB controller
|
|
||||||
E: ID_PCI_INTERFACE_FROM_DATABASE=XHCI
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Intel Corporation
|
|
||||||
E: ID_MODEL_FROM_DATABASE=Cannon Point-LP USB 3.1 xHCI Controller
|
|
||||||
A: ari_enabled=0
|
|
||||||
A: broken_parity_status=0
|
|
||||||
A: class=0x0c0330
|
|
||||||
H: config=8680ED9D060490023030030C00008000040030A10000000000000000000000000000000000000000000000003C10EF85000000007000000000000000FF010000FD0134808FC6FF8300000000000000007F6DDC0F00000000181C030400000000316000000000000000000000000000000180C2C1080000000000000000000000059087007802E0FE0000000000000000090014F01000400100000000C10A080000080E00001800008F40020000010000010000000000000000000000000000000000000000000000000000000000000001000000000000000000000000000000000000000000000000000000000000000000000000000000B50F300112000000
|
|
||||||
A: consistent_dma_mask_bits=64
|
|
||||||
A: d3cold_allowed=1
|
|
||||||
A: dbc=disabled
|
|
||||||
A: device=0x9ded
|
|
||||||
A: dma_mask_bits=64
|
|
||||||
L: driver=../../../bus/pci/drivers/xhci_hcd
|
|
||||||
A: driver_override=(null)
|
|
||||||
A: enable=1
|
|
||||||
A: irq=124
|
|
||||||
A: local_cpulist=0-3
|
|
||||||
A: local_cpus=f
|
|
||||||
A: modalias=pci:v00008086d00009DEDsv0000103Csd000085EFbc0Csc03i30
|
|
||||||
A: msi_bus=1
|
|
||||||
A: msi_irqs/124=msi
|
|
||||||
A: numa_node=-1
|
|
||||||
A: pools=poolinfo - 0.1\nbuffer-2048 0 0 2048 0\nbuffer-512 0 0 512 0\nbuffer-128 0 32 128 1\nbuffer-32 0 0 32 0\nxHCI 1KB stream ctx arrays 0 0 1024 0\nxHCI 256 byte stream ctx arrays 0 0 256 0\nxHCI input/output contexts 11 12 2112 12\nxHCI ring segments 54 54 4096 54\nbuffer-2048 0 0 2048 0\nbuffer-512 0 0 512 0\nbuffer-128 9 32 128 1\nbuffer-32 0 0 32 0
|
|
||||||
A: power/async=enabled
|
|
||||||
A: power/control=auto
|
|
||||||
A: power/runtime_active_kids=1
|
|
||||||
A: power/runtime_active_time=74606194
|
|
||||||
A: power/runtime_enabled=enabled
|
|
||||||
A: power/runtime_status=active
|
|
||||||
A: power/runtime_suspended_time=0
|
|
||||||
A: power/runtime_usage=0
|
|
||||||
A: power/wakeup=enabled
|
|
||||||
A: power/wakeup_abort_count=0
|
|
||||||
A: power/wakeup_active=0
|
|
||||||
A: power/wakeup_active_count=0
|
|
||||||
A: power/wakeup_count=0
|
|
||||||
A: power/wakeup_expire_count=0
|
|
||||||
A: power/wakeup_last_time_ms=0
|
|
||||||
A: power/wakeup_max_time_ms=0
|
|
||||||
A: power/wakeup_total_time_ms=0
|
|
||||||
A: resource=0x00000000a1300000 0x00000000a130ffff 0x0000000000140204\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000
|
|
||||||
A: revision=0x30
|
|
||||||
A: subsystem_device=0x85ef
|
|
||||||
A: subsystem_vendor=0x103c
|
|
||||||
A: vendor=0x8086
|
|
||||||
|
|
|
@ -12,6 +12,9 @@ envs.prepend('LD_LIBRARY_PATH', join_paths(meson.build_root(), 'libfprint'))
|
||||||
# random numbers rather than proper ones)
|
# random numbers rather than proper ones)
|
||||||
envs.set('FP_DEVICE_EMULATION', '1')
|
envs.set('FP_DEVICE_EMULATION', '1')
|
||||||
|
|
||||||
|
# Path to SDCP virtual device binary, only used for virtual-sdcp test
|
||||||
|
envs.set('SDCP_VIRT_BINARY', get_option('sdcp_virt_binary'))
|
||||||
|
|
||||||
# Set a colon-separated list of native drivers we enable in tests
|
# Set a colon-separated list of native drivers we enable in tests
|
||||||
envs.set('FP_DRIVERS_WHITELIST', ':'.join([
|
envs.set('FP_DRIVERS_WHITELIST', ':'.join([
|
||||||
'virtual_image',
|
'virtual_image',
|
||||||
|
@ -22,34 +25,18 @@ envs.set('FP_DRIVERS_WHITELIST', ':'.join([
|
||||||
envs.set('NO_AT_BRIDGE', '1')
|
envs.set('NO_AT_BRIDGE', '1')
|
||||||
|
|
||||||
drivers_tests = [
|
drivers_tests = [
|
||||||
'aes2501',
|
|
||||||
'aes3500',
|
'aes3500',
|
||||||
'elan',
|
'elan',
|
||||||
'elan-cobo',
|
|
||||||
'elanmoc',
|
|
||||||
'elanspi',
|
'elanspi',
|
||||||
'synaptics',
|
'synaptics',
|
||||||
'upektc_img',
|
|
||||||
'uru4000-msv2',
|
|
||||||
'uru4000-4500',
|
|
||||||
'vfs0050',
|
'vfs0050',
|
||||||
'vfs301',
|
'vfs301',
|
||||||
'vfs5011',
|
'vfs5011',
|
||||||
'vfs7552',
|
'vfs7552',
|
||||||
'goodixmoc',
|
'goodixmoc',
|
||||||
'nb1010',
|
'nb1010',
|
||||||
'egis0570',
|
|
||||||
]
|
]
|
||||||
|
|
||||||
if get_option('introspection')
|
|
||||||
conf = configuration_data()
|
|
||||||
conf.set('SRCDIR', meson.source_root())
|
|
||||||
conf.set('BUILDDIR', meson.build_root())
|
|
||||||
configure_file(configuration: conf,
|
|
||||||
input: 'create-driver-test.py.in',
|
|
||||||
output: 'create-driver-test.py')
|
|
||||||
endif
|
|
||||||
|
|
||||||
if get_option('introspection')
|
if get_option('introspection')
|
||||||
envs.prepend('GI_TYPELIB_PATH', join_paths(meson.build_root(), 'libfprint'))
|
envs.prepend('GI_TYPELIB_PATH', join_paths(meson.build_root(), 'libfprint'))
|
||||||
virtual_devices_tests = [
|
virtual_devices_tests = [
|
||||||
|
@ -88,6 +75,7 @@ if get_option('introspection')
|
||||||
suite: ut_suite,
|
suite: ut_suite,
|
||||||
depends: libfprint_typelib,
|
depends: libfprint_typelib,
|
||||||
env: envs,
|
env: envs,
|
||||||
|
workdir: meson.current_source_dir(),
|
||||||
)
|
)
|
||||||
endforeach
|
endforeach
|
||||||
else
|
else
|
||||||
|
@ -98,8 +86,52 @@ if get_option('introspection')
|
||||||
endif
|
endif
|
||||||
endforeach
|
endforeach
|
||||||
|
|
||||||
|
if 'virtual_sdcp' in drivers and get_option('sdcp_virt_binary') != ''
|
||||||
|
python3 = find_program('python3')
|
||||||
|
unittest_inspector = find_program('unittest_inspector.py')
|
||||||
|
base_args = files('virtual-sdcp.py')
|
||||||
|
suite = []
|
||||||
|
|
||||||
|
r = run_command(unittest_inspector, files('virtual-sdcp.py'))
|
||||||
|
unit_tests = r.stdout().strip().split('\n')
|
||||||
|
|
||||||
|
if r.returncode() == 0 and unit_tests.length() > 0
|
||||||
|
suite += 'virtual-sdcp'
|
||||||
|
else
|
||||||
|
unit_tests = ['virtual-sdcp']
|
||||||
|
endif
|
||||||
|
|
||||||
|
sdcp_envs = envs
|
||||||
|
sdcp_envs.set('FP_DEVICE_EMULATION', '0')
|
||||||
|
foreach ut: unit_tests
|
||||||
|
ut_suite = suite
|
||||||
|
ut_args = base_args
|
||||||
|
if unit_tests.length() > 1
|
||||||
|
ut_args += ut
|
||||||
|
ut_suite += ut.split('.')[0]
|
||||||
|
endif
|
||||||
|
test(ut,
|
||||||
|
python3,
|
||||||
|
args: ut_args,
|
||||||
|
suite: ut_suite,
|
||||||
|
depends: libfprint_typelib,
|
||||||
|
env: sdcp_envs,
|
||||||
|
workdir: meson.current_source_dir(),
|
||||||
|
)
|
||||||
|
endforeach
|
||||||
|
else
|
||||||
|
test('virtual-sdcp',
|
||||||
|
find_program('sh'),
|
||||||
|
args: ['-c', 'exit 77']
|
||||||
|
)
|
||||||
|
endif
|
||||||
|
|
||||||
foreach driver_test: drivers_tests
|
foreach driver_test: drivers_tests
|
||||||
|
if driver_test.contains('-')
|
||||||
driver_name = driver_test.split('-')[0]
|
driver_name = driver_test.split('-')[0]
|
||||||
|
else
|
||||||
|
driver_name = driver_test
|
||||||
|
endif
|
||||||
driver_envs = envs
|
driver_envs = envs
|
||||||
driver_envs.set('FP_DRIVERS_WHITELIST', driver_name)
|
driver_envs.set('FP_DRIVERS_WHITELIST', driver_name)
|
||||||
|
|
||||||
|
@ -109,6 +141,7 @@ if get_option('introspection')
|
||||||
find_program('umockdev-test.py'),
|
find_program('umockdev-test.py'),
|
||||||
args: join_paths(meson.current_source_dir(), driver_test),
|
args: join_paths(meson.current_source_dir(), driver_test),
|
||||||
env: driver_envs,
|
env: driver_envs,
|
||||||
|
workdir: meson.current_source_dir(),
|
||||||
suite: ['drivers'],
|
suite: ['drivers'],
|
||||||
timeout: 15,
|
timeout: 15,
|
||||||
depends: libfprint_typelib,
|
depends: libfprint_typelib,
|
||||||
|
|
BIN
tests/sdcp-key-db/cert9.db
Normal file
BIN
tests/sdcp-key-db/cert9.db
Normal file
Binary file not shown.
BIN
tests/sdcp-key-db/key4.db
Normal file
BIN
tests/sdcp-key-db/key4.db
Normal file
Binary file not shown.
|
@ -271,26 +271,6 @@ fpi_device_fake_cancel (FpDevice *device)
|
||||||
g_assert_cmpuint (fpi_device_get_current_action (device), !=, FPI_DEVICE_ACTION_NONE);
|
g_assert_cmpuint (fpi_device_get_current_action (device), !=, FPI_DEVICE_ACTION_NONE);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
fpi_device_fake_suspend (FpDevice *device)
|
|
||||||
{
|
|
||||||
FpiDeviceFake *fake_dev = FPI_DEVICE_FAKE (device);
|
|
||||||
|
|
||||||
fake_dev->last_called_function = fpi_device_fake_suspend;
|
|
||||||
|
|
||||||
fpi_device_suspend_complete (device, g_steal_pointer (&fake_dev->ret_suspend));
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
fpi_device_fake_resume (FpDevice *device)
|
|
||||||
{
|
|
||||||
FpiDeviceFake *fake_dev = FPI_DEVICE_FAKE (device);
|
|
||||||
|
|
||||||
fake_dev->last_called_function = fpi_device_fake_resume;
|
|
||||||
|
|
||||||
fpi_device_resume_complete (device, g_steal_pointer (&fake_dev->ret_resume));
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
fpi_device_fake_init (FpiDeviceFake *self)
|
fpi_device_fake_init (FpiDeviceFake *self)
|
||||||
{
|
{
|
||||||
|
@ -319,8 +299,6 @@ fpi_device_fake_class_init (FpiDeviceFakeClass *klass)
|
||||||
dev_class->delete = fpi_device_fake_delete;
|
dev_class->delete = fpi_device_fake_delete;
|
||||||
dev_class->cancel = fpi_device_fake_cancel;
|
dev_class->cancel = fpi_device_fake_cancel;
|
||||||
dev_class->clear_storage = fpi_device_fake_clear_storage;
|
dev_class->clear_storage = fpi_device_fake_clear_storage;
|
||||||
dev_class->suspend = fpi_device_fake_suspend;
|
|
||||||
dev_class->resume = fpi_device_fake_resume;
|
|
||||||
|
|
||||||
fpi_device_class_auto_initialize_features (dev_class);
|
fpi_device_class_auto_initialize_features (dev_class);
|
||||||
}
|
}
|
||||||
|
|
|
@ -32,8 +32,6 @@ struct _FpiDeviceFake
|
||||||
gpointer last_called_function;
|
gpointer last_called_function;
|
||||||
gboolean return_action_error;
|
gboolean return_action_error;
|
||||||
|
|
||||||
GCancellable *ext_cancellable;
|
|
||||||
|
|
||||||
GError *ret_error;
|
GError *ret_error;
|
||||||
FpPrint *ret_print;
|
FpPrint *ret_print;
|
||||||
FpPrint *ret_match;
|
FpPrint *ret_match;
|
||||||
|
@ -41,9 +39,6 @@ struct _FpiDeviceFake
|
||||||
FpImage *ret_image;
|
FpImage *ret_image;
|
||||||
GPtrArray *ret_list;
|
GPtrArray *ret_list;
|
||||||
|
|
||||||
GError *ret_suspend;
|
|
||||||
GError *ret_resume;
|
|
||||||
|
|
||||||
gpointer action_data;
|
gpointer action_data;
|
||||||
gpointer user_data;
|
gpointer user_data;
|
||||||
|
|
||||||
|
|
|
@ -35,11 +35,9 @@ typedef FpDevice FpAutoCloseDevice;
|
||||||
static FpAutoCloseDevice *
|
static FpAutoCloseDevice *
|
||||||
auto_close_fake_device_new (void)
|
auto_close_fake_device_new (void)
|
||||||
{
|
{
|
||||||
g_autoptr(GError) error = NULL;
|
|
||||||
FpAutoCloseDevice *device = g_object_new (FPI_TYPE_DEVICE_FAKE, NULL);
|
FpAutoCloseDevice *device = g_object_new (FPI_TYPE_DEVICE_FAKE, NULL);
|
||||||
|
|
||||||
if (!fp_device_open_sync (device, NULL, &error))
|
g_assert_true (fp_device_open_sync (device, NULL, NULL));
|
||||||
g_error ("Could not open device: %s", error->message);
|
|
||||||
|
|
||||||
return device;
|
return device;
|
||||||
}
|
}
|
||||||
|
@ -47,7 +45,6 @@ auto_close_fake_device_new (void)
|
||||||
static void
|
static void
|
||||||
auto_close_fake_device_free (FpAutoCloseDevice *device)
|
auto_close_fake_device_free (FpAutoCloseDevice *device)
|
||||||
{
|
{
|
||||||
g_autoptr(GError) error = NULL;
|
|
||||||
FpiDeviceFake *fake_dev = FPI_DEVICE_FAKE (device);
|
FpiDeviceFake *fake_dev = FPI_DEVICE_FAKE (device);
|
||||||
|
|
||||||
if (fake_dev->return_action_error)
|
if (fake_dev->return_action_error)
|
||||||
|
@ -57,8 +54,7 @@ auto_close_fake_device_free (FpAutoCloseDevice *device)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (fp_device_is_open (device))
|
if (fp_device_is_open (device))
|
||||||
if (!fp_device_close_sync (device, NULL, &error))
|
g_assert_true (fp_device_close_sync (device, NULL, NULL));
|
||||||
g_error ("Could not close device: %s", error->message);
|
|
||||||
|
|
||||||
g_object_unref (device);
|
g_object_unref (device);
|
||||||
}
|
}
|
||||||
|
@ -1228,11 +1224,6 @@ test_driver_match_cb (FpDevice *device,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
fake_device_stub_verify (FpDevice *device)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
test_driver_verify (void)
|
test_driver_verify (void)
|
||||||
{
|
{
|
||||||
|
@ -1597,28 +1588,6 @@ fake_device_stub_identify (FpDevice *device)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
test_driver_identify_cb (FpDevice *device,
|
|
||||||
GAsyncResult *res,
|
|
||||||
gpointer user_data)
|
|
||||||
{
|
|
||||||
MatchCbData *data = user_data;
|
|
||||||
gboolean r;
|
|
||||||
|
|
||||||
g_assert (data->called == FALSE);
|
|
||||||
data->called = TRUE;
|
|
||||||
|
|
||||||
r = fp_device_identify_finish (device, res, &data->match, &data->print, &data->error);
|
|
||||||
|
|
||||||
if (r)
|
|
||||||
g_assert_no_error (data->error);
|
|
||||||
else
|
|
||||||
g_assert_nonnull (data->error);
|
|
||||||
|
|
||||||
if (data->match)
|
|
||||||
g_assert_no_error (data->error);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
test_driver_supports_identify (void)
|
test_driver_supports_identify (void)
|
||||||
{
|
{
|
||||||
|
@ -1972,314 +1941,6 @@ test_driver_identify_report_no_callback (void)
|
||||||
g_assert_false (match);
|
g_assert_false (match);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
test_driver_identify_suspend_continues (void)
|
|
||||||
{
|
|
||||||
g_autoptr(FpAutoResetClass) dev_class = auto_reset_device_class ();
|
|
||||||
g_autoptr(MatchCbData) match_data = g_new0 (MatchCbData, 1);
|
|
||||||
g_autoptr(MatchCbData) identify_data = g_new0 (MatchCbData, 1);
|
|
||||||
g_autoptr(GPtrArray) prints = NULL;
|
|
||||||
g_autoptr(FpAutoCloseDevice) device = NULL;
|
|
||||||
g_autoptr(GError) error = NULL;
|
|
||||||
void (*orig_identify) (FpDevice *device);
|
|
||||||
FpiDeviceFake *fake_dev;
|
|
||||||
FpPrint *expected_matched;
|
|
||||||
|
|
||||||
device = g_object_new (FPI_TYPE_DEVICE_FAKE, NULL);
|
|
||||||
fake_dev = FPI_DEVICE_FAKE (device);
|
|
||||||
orig_identify = dev_class->identify;
|
|
||||||
dev_class->identify = fake_device_stub_identify;
|
|
||||||
|
|
||||||
prints = make_fake_prints_gallery (device, 500);
|
|
||||||
expected_matched = g_ptr_array_index (prints, g_random_int_range (0, 499));
|
|
||||||
fp_print_set_description (expected_matched, "fake-verified");
|
|
||||||
|
|
||||||
match_data->gallery = prints;
|
|
||||||
|
|
||||||
fake_dev->ret_print = make_fake_print (device, NULL);
|
|
||||||
|
|
||||||
g_assert_true (fp_device_open_sync (device, NULL, NULL));
|
|
||||||
|
|
||||||
fp_device_identify (device, prints, NULL,
|
|
||||||
test_driver_match_cb, match_data, NULL,
|
|
||||||
(GAsyncReadyCallback) test_driver_identify_cb, identify_data);
|
|
||||||
|
|
||||||
while (g_main_context_iteration (NULL, FALSE))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
fake_dev->ret_suspend = NULL;
|
|
||||||
fp_device_suspend_sync (device, NULL, &error);
|
|
||||||
g_assert (fake_dev->last_called_function == dev_class->suspend);
|
|
||||||
g_assert_no_error (error);
|
|
||||||
|
|
||||||
while (g_main_context_iteration (NULL, FALSE))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
g_assert_false (match_data->called);
|
|
||||||
g_assert_false (identify_data->called);
|
|
||||||
|
|
||||||
fake_dev->ret_resume = NULL;
|
|
||||||
fp_device_resume_sync (device, NULL, &error);
|
|
||||||
g_assert (fake_dev->last_called_function == dev_class->resume);
|
|
||||||
g_assert_no_error (error);
|
|
||||||
|
|
||||||
orig_identify (device);
|
|
||||||
|
|
||||||
/* This currently happens immediately (not ABI though) */
|
|
||||||
g_assert_true (match_data->called);
|
|
||||||
g_assert (match_data->match == expected_matched);
|
|
||||||
|
|
||||||
while (g_main_context_iteration (NULL, FALSE))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
g_assert_true (identify_data->called);
|
|
||||||
g_assert (identify_data->match == expected_matched);
|
|
||||||
|
|
||||||
g_assert (fake_dev->last_called_function == orig_identify);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
test_driver_identify_suspend_succeeds (void)
|
|
||||||
{
|
|
||||||
g_autoptr(FpAutoResetClass) dev_class = auto_reset_device_class ();
|
|
||||||
g_autoptr(MatchCbData) match_data = g_new0 (MatchCbData, 1);
|
|
||||||
g_autoptr(MatchCbData) identify_data = g_new0 (MatchCbData, 1);
|
|
||||||
g_autoptr(GPtrArray) prints = NULL;
|
|
||||||
g_autoptr(FpAutoCloseDevice) device = NULL;
|
|
||||||
g_autoptr(GError) error = NULL;
|
|
||||||
void (*orig_identify) (FpDevice *device);
|
|
||||||
FpiDeviceFake *fake_dev;
|
|
||||||
FpPrint *expected_matched;
|
|
||||||
|
|
||||||
device = g_object_new (FPI_TYPE_DEVICE_FAKE, NULL);
|
|
||||||
fake_dev = FPI_DEVICE_FAKE (device);
|
|
||||||
orig_identify = dev_class->identify;
|
|
||||||
dev_class->identify = fake_device_stub_identify;
|
|
||||||
|
|
||||||
prints = make_fake_prints_gallery (device, 500);
|
|
||||||
expected_matched = g_ptr_array_index (prints, g_random_int_range (0, 499));
|
|
||||||
fp_print_set_description (expected_matched, "fake-verified");
|
|
||||||
|
|
||||||
match_data->gallery = prints;
|
|
||||||
|
|
||||||
g_assert_true (fp_device_open_sync (device, NULL, NULL));
|
|
||||||
|
|
||||||
fake_dev->ret_print = make_fake_print (device, NULL);
|
|
||||||
fp_device_identify (device, prints, NULL,
|
|
||||||
test_driver_match_cb, match_data, NULL,
|
|
||||||
(GAsyncReadyCallback) test_driver_identify_cb, identify_data);
|
|
||||||
|
|
||||||
while (g_main_context_iteration (NULL, FALSE))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
/* suspend_sync hangs until cancellation, so we need to trigger orig_identify
|
|
||||||
* from the mainloop after calling suspend_sync.
|
|
||||||
*/
|
|
||||||
fpi_device_add_timeout (device, 0, (FpTimeoutFunc) orig_identify, NULL, NULL);
|
|
||||||
|
|
||||||
fake_dev->ret_suspend = fpi_device_error_new (FP_DEVICE_ERROR_NOT_SUPPORTED);
|
|
||||||
fp_device_suspend_sync (device, NULL, &error);
|
|
||||||
|
|
||||||
/* At this point we are done with everything */
|
|
||||||
g_assert (fake_dev->last_called_function == orig_identify);
|
|
||||||
g_assert_error (error, FP_DEVICE_ERROR, FP_DEVICE_ERROR_NOT_SUPPORTED);
|
|
||||||
g_clear_error (&error);
|
|
||||||
|
|
||||||
/* We suspended, but device reported success and that will be reported. */
|
|
||||||
g_assert_true (match_data->called);
|
|
||||||
g_assert (match_data->match == expected_matched);
|
|
||||||
g_assert_true (identify_data->called);
|
|
||||||
g_assert (identify_data->match == expected_matched);
|
|
||||||
|
|
||||||
/* Resuming the device does not call resume handler, as the action was
|
|
||||||
* cancelled already.
|
|
||||||
*/
|
|
||||||
fake_dev->last_called_function = NULL;
|
|
||||||
fp_device_resume_sync (device, NULL, &error);
|
|
||||||
g_assert (fake_dev->last_called_function == NULL);
|
|
||||||
g_assert_no_error (error);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
test_driver_identify_suspend_busy_error (void)
|
|
||||||
{
|
|
||||||
g_autoptr(FpAutoResetClass) dev_class = auto_reset_device_class ();
|
|
||||||
g_autoptr(MatchCbData) match_data = g_new0 (MatchCbData, 1);
|
|
||||||
g_autoptr(MatchCbData) identify_data = g_new0 (MatchCbData, 1);
|
|
||||||
g_autoptr(GPtrArray) prints = NULL;
|
|
||||||
g_autoptr(FpAutoCloseDevice) device = NULL;
|
|
||||||
g_autoptr(GError) error = NULL;
|
|
||||||
void (*orig_identify) (FpDevice *device);
|
|
||||||
FpiDeviceFake *fake_dev;
|
|
||||||
FpPrint *expected_matched;
|
|
||||||
|
|
||||||
device = g_object_new (FPI_TYPE_DEVICE_FAKE, NULL);
|
|
||||||
fake_dev = FPI_DEVICE_FAKE (device);
|
|
||||||
orig_identify = dev_class->identify;
|
|
||||||
dev_class->identify = fake_device_stub_identify;
|
|
||||||
|
|
||||||
prints = make_fake_prints_gallery (device, 500);
|
|
||||||
expected_matched = g_ptr_array_index (prints, g_random_int_range (0, 499));
|
|
||||||
fp_print_set_description (expected_matched, "fake-verified");
|
|
||||||
|
|
||||||
match_data->gallery = prints;
|
|
||||||
|
|
||||||
g_assert_true (fp_device_open_sync (device, NULL, NULL));
|
|
||||||
|
|
||||||
fake_dev->ret_error = fpi_device_error_new (FP_DEVICE_ERROR_GENERAL);
|
|
||||||
fake_dev->ret_print = make_fake_print (device, NULL);
|
|
||||||
fp_device_identify (device, prints, NULL,
|
|
||||||
test_driver_match_cb, match_data, NULL,
|
|
||||||
(GAsyncReadyCallback) test_driver_identify_cb, identify_data);
|
|
||||||
|
|
||||||
while (g_main_context_iteration (NULL, FALSE))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
/* suspend_sync hangs until cancellation, so we need to trigger orig_identify
|
|
||||||
* from the mainloop after calling suspend_sync.
|
|
||||||
*/
|
|
||||||
fpi_device_add_timeout (device, 0, (FpTimeoutFunc) orig_identify, NULL, NULL);
|
|
||||||
|
|
||||||
fake_dev->ret_suspend = fpi_device_error_new (FP_DEVICE_ERROR_NOT_SUPPORTED);
|
|
||||||
fp_device_suspend_sync (device, NULL, &error);
|
|
||||||
fake_dev->ret_error = NULL;
|
|
||||||
|
|
||||||
/* At this point we are done with everything */
|
|
||||||
g_assert (fake_dev->last_called_function == orig_identify);
|
|
||||||
g_assert_error (error, FP_DEVICE_ERROR, FP_DEVICE_ERROR_NOT_SUPPORTED);
|
|
||||||
g_clear_error (&error);
|
|
||||||
|
|
||||||
/* The device reported an error, an this error will be overwritten.
|
|
||||||
*/
|
|
||||||
g_assert_false (match_data->called);
|
|
||||||
g_assert_true (identify_data->called);
|
|
||||||
g_assert_null (identify_data->match);
|
|
||||||
g_assert_error (identify_data->error, FP_DEVICE_ERROR, FP_DEVICE_ERROR_BUSY);
|
|
||||||
|
|
||||||
fake_dev->last_called_function = NULL;
|
|
||||||
fp_device_resume_sync (device, NULL, &error);
|
|
||||||
g_assert (fake_dev->last_called_function == NULL);
|
|
||||||
g_assert_no_error (error);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
test_driver_identify_suspend_while_idle (void)
|
|
||||||
{
|
|
||||||
g_autoptr(FpAutoResetClass) dev_class = auto_reset_device_class ();
|
|
||||||
g_autoptr(FpAutoCloseDevice) device = NULL;
|
|
||||||
g_autoptr(GError) error = NULL;
|
|
||||||
FpiDeviceFake *fake_dev;
|
|
||||||
|
|
||||||
device = g_object_new (FPI_TYPE_DEVICE_FAKE, NULL);
|
|
||||||
fake_dev = FPI_DEVICE_FAKE (device);
|
|
||||||
|
|
||||||
/* Suspending and resuming a closed device works */
|
|
||||||
fp_device_suspend (device, NULL, (GAsyncReadyCallback) fp_device_suspend_finish, &error);
|
|
||||||
while (g_main_context_iteration (NULL, FALSE))
|
|
||||||
continue;
|
|
||||||
g_assert (fake_dev->last_called_function == NULL);
|
|
||||||
g_assert_no_error (error);
|
|
||||||
|
|
||||||
fp_device_resume (device, NULL, (GAsyncReadyCallback) fp_device_resume_finish, NULL);
|
|
||||||
while (g_main_context_iteration (NULL, FALSE))
|
|
||||||
continue;
|
|
||||||
g_assert (fake_dev->last_called_function == NULL);
|
|
||||||
g_assert_no_error (error);
|
|
||||||
|
|
||||||
g_assert_true (fp_device_open_sync (device, NULL, NULL));
|
|
||||||
|
|
||||||
fake_dev->last_called_function = NULL;
|
|
||||||
fp_device_suspend (device, NULL, (GAsyncReadyCallback) fp_device_suspend_finish, &error);
|
|
||||||
while (g_main_context_iteration (NULL, FALSE))
|
|
||||||
continue;
|
|
||||||
g_assert (fake_dev->last_called_function == NULL);
|
|
||||||
g_assert_no_error (error);
|
|
||||||
|
|
||||||
fp_device_resume (device, NULL, (GAsyncReadyCallback) fp_device_resume_finish, NULL);
|
|
||||||
while (g_main_context_iteration (NULL, FALSE))
|
|
||||||
continue;
|
|
||||||
g_assert (fake_dev->last_called_function == NULL);
|
|
||||||
g_assert_no_error (error);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
test_driver_identify_warmup_cooldown (void)
|
|
||||||
{
|
|
||||||
g_autoptr(FpAutoResetClass) dev_class = auto_reset_device_class ();
|
|
||||||
g_autoptr(MatchCbData) identify_data = g_new0 (MatchCbData, 1);
|
|
||||||
g_autoptr(GPtrArray) prints = NULL;
|
|
||||||
g_autoptr(FpAutoCloseDevice) device = NULL;
|
|
||||||
g_autoptr(GError) error = NULL;
|
|
||||||
void (*orig_identify) (FpDevice *device);
|
|
||||||
FpiDeviceFake *fake_dev;
|
|
||||||
gint64 start_time;
|
|
||||||
|
|
||||||
dev_class->temp_hot_seconds = 2;
|
|
||||||
dev_class->temp_cold_seconds = 5;
|
|
||||||
|
|
||||||
device = g_object_new (FPI_TYPE_DEVICE_FAKE, NULL);
|
|
||||||
fake_dev = FPI_DEVICE_FAKE (device);
|
|
||||||
orig_identify = dev_class->identify;
|
|
||||||
dev_class->identify = fake_device_stub_identify;
|
|
||||||
|
|
||||||
prints = make_fake_prints_gallery (device, 500);
|
|
||||||
|
|
||||||
g_assert_true (fp_device_open_sync (device, NULL, NULL));
|
|
||||||
fake_dev->last_called_function = NULL;
|
|
||||||
|
|
||||||
fake_dev->ret_error = fpi_device_error_new (FP_DEVICE_ERROR_GENERAL);
|
|
||||||
|
|
||||||
/* Undefined: Whether match_cb is called. */
|
|
||||||
fp_device_identify (device, prints, NULL,
|
|
||||||
NULL, NULL, NULL,
|
|
||||||
(GAsyncReadyCallback) test_driver_identify_cb, identify_data);
|
|
||||||
|
|
||||||
/* Identify is running, the temperature will change after only a short time.
|
|
||||||
* Changes are delayed by 100ms and we give 150ms of slack for the test.
|
|
||||||
*/
|
|
||||||
start_time = g_get_monotonic_time ();
|
|
||||||
g_assert_cmpint (fp_device_get_temperature (device), ==, FP_TEMPERATURE_COLD);
|
|
||||||
while (fp_device_get_temperature (device) == FP_TEMPERATURE_COLD)
|
|
||||||
g_main_context_iteration (NULL, TRUE);
|
|
||||||
g_assert_cmpint (fp_device_get_temperature (device), ==, FP_TEMPERATURE_WARM);
|
|
||||||
g_assert_false (g_cancellable_is_cancelled (fpi_device_get_cancellable (device)));
|
|
||||||
g_assert_cmpint (g_get_monotonic_time () - start_time, <, 0 + 250000);
|
|
||||||
|
|
||||||
/* we reach hot 2 seconds later */
|
|
||||||
while (fp_device_get_temperature (device) == FP_TEMPERATURE_WARM)
|
|
||||||
g_main_context_iteration (NULL, TRUE);
|
|
||||||
g_assert_cmpint (fp_device_get_temperature (device), ==, FP_TEMPERATURE_HOT);
|
|
||||||
g_assert_true (g_cancellable_is_cancelled (fpi_device_get_cancellable (device)));
|
|
||||||
g_assert_cmpint (g_get_monotonic_time () - start_time, <, 2000000 + 250000);
|
|
||||||
|
|
||||||
/* cancel vfunc will be called now */
|
|
||||||
g_assert (fake_dev->last_called_function == NULL);
|
|
||||||
while (g_main_context_iteration (NULL, FALSE))
|
|
||||||
continue;
|
|
||||||
g_assert (fake_dev->last_called_function == dev_class->cancel);
|
|
||||||
|
|
||||||
orig_identify (device);
|
|
||||||
fake_dev->ret_error = NULL;
|
|
||||||
while (g_main_context_iteration (NULL, FALSE))
|
|
||||||
continue;
|
|
||||||
g_assert_true (identify_data->called);
|
|
||||||
g_assert_error (identify_data->error, FP_DEVICE_ERROR, FP_DEVICE_ERROR_TOO_HOT);
|
|
||||||
|
|
||||||
/* Now, wait for it to cool down again;
|
|
||||||
* WARM should be reached after about 2s
|
|
||||||
* COLD after 5s but give it some more slack. */
|
|
||||||
start_time = g_get_monotonic_time ();
|
|
||||||
while (fp_device_get_temperature (device) == FP_TEMPERATURE_HOT)
|
|
||||||
g_main_context_iteration (NULL, TRUE);
|
|
||||||
g_assert_cmpint (fp_device_get_temperature (device), ==, FP_TEMPERATURE_WARM);
|
|
||||||
g_assert_cmpint (g_get_monotonic_time () - start_time, <, 2000000 + 250000);
|
|
||||||
|
|
||||||
while (fp_device_get_temperature (device) == FP_TEMPERATURE_WARM)
|
|
||||||
g_main_context_iteration (NULL, TRUE);
|
|
||||||
g_assert_cmpint (fp_device_get_temperature (device), ==, FP_TEMPERATURE_COLD);
|
|
||||||
g_assert_cmpint (g_get_monotonic_time () - start_time, <, 5000000 + 500000);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
fake_device_stub_capture (FpDevice *device)
|
fake_device_stub_capture (FpDevice *device)
|
||||||
{
|
{
|
||||||
|
@ -2623,89 +2284,6 @@ test_driver_cancel_fail (void)
|
||||||
g_assert_no_error (error);
|
g_assert_no_error (error);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
test_driver_critical (void)
|
|
||||||
{
|
|
||||||
g_autoptr(FpAutoCloseDevice) device = auto_close_fake_device_new ();
|
|
||||||
g_autoptr(GCancellable) cancellable = g_cancellable_new ();
|
|
||||||
g_autoptr(FpPrint) enrolled_print = make_fake_print_reffed (device, NULL);
|
|
||||||
g_autoptr(FpAutoResetClass) dev_class = auto_reset_device_class ();
|
|
||||||
void (*orig_verify) (FpDevice *device) = dev_class->verify;
|
|
||||||
FpiDeviceFake *fake_dev = FPI_DEVICE_FAKE (device);
|
|
||||||
|
|
||||||
fake_dev->last_called_function = NULL;
|
|
||||||
|
|
||||||
dev_class->verify = fake_device_stub_verify;
|
|
||||||
fp_device_verify (device, enrolled_print, cancellable,
|
|
||||||
NULL, NULL, NULL,
|
|
||||||
NULL, NULL);
|
|
||||||
|
|
||||||
/* We started a verify operation, now emulate a "critical" section */
|
|
||||||
fpi_device_critical_enter (device);
|
|
||||||
|
|
||||||
/* Throw a suspend and external cancellation against it. */
|
|
||||||
fp_device_suspend (device, NULL, NULL, NULL);
|
|
||||||
g_cancellable_cancel (cancellable);
|
|
||||||
|
|
||||||
/* The only thing that happens is that the cancellable is cancelled */
|
|
||||||
g_assert_true (fpi_device_action_is_cancelled (device));
|
|
||||||
g_assert (fake_dev->last_called_function == NULL);
|
|
||||||
while (g_main_context_iteration (NULL, FALSE))
|
|
||||||
continue;
|
|
||||||
g_assert (fake_dev->last_called_function == NULL);
|
|
||||||
|
|
||||||
/* Leaving and entering the critical section in the same mainloop iteration
|
|
||||||
* does not do anything. */
|
|
||||||
fpi_device_critical_leave (device);
|
|
||||||
fpi_device_critical_enter (device);
|
|
||||||
while (g_main_context_iteration (NULL, FALSE))
|
|
||||||
continue;
|
|
||||||
g_assert (fake_dev->last_called_function == NULL);
|
|
||||||
|
|
||||||
/* Leaving it and running the mainloop will first run the cancel handler */
|
|
||||||
fpi_device_critical_leave (device);
|
|
||||||
while (g_main_context_iteration (NULL, FALSE) && !fake_dev->last_called_function)
|
|
||||||
continue;
|
|
||||||
g_assert (fake_dev->last_called_function == dev_class->cancel);
|
|
||||||
g_assert_true (fpi_device_action_is_cancelled (device));
|
|
||||||
fake_dev->last_called_function = NULL;
|
|
||||||
|
|
||||||
/* Then the suspend handler */
|
|
||||||
while (g_main_context_iteration (NULL, FALSE) && !fake_dev->last_called_function)
|
|
||||||
continue;
|
|
||||||
g_assert (fake_dev->last_called_function == dev_class->suspend);
|
|
||||||
fake_dev->last_called_function = NULL;
|
|
||||||
|
|
||||||
/* Nothing happens afterwards */
|
|
||||||
while (g_main_context_iteration (NULL, FALSE))
|
|
||||||
continue;
|
|
||||||
g_assert (fake_dev->last_called_function == NULL);
|
|
||||||
|
|
||||||
|
|
||||||
/* Throw a resume at the system */
|
|
||||||
fpi_device_critical_enter (device);
|
|
||||||
fp_device_resume (device, NULL, NULL, NULL);
|
|
||||||
|
|
||||||
/* Nothing will happen, as the resume is delayed */
|
|
||||||
while (g_main_context_iteration (NULL, FALSE))
|
|
||||||
continue;
|
|
||||||
g_assert (fake_dev->last_called_function == NULL);
|
|
||||||
|
|
||||||
/* Finally the resume is called from the mainloop after leaving the critical section */
|
|
||||||
fpi_device_critical_leave (device);
|
|
||||||
g_assert (fake_dev->last_called_function == NULL);
|
|
||||||
while (g_main_context_iteration (NULL, FALSE) && !fake_dev->last_called_function)
|
|
||||||
continue;
|
|
||||||
g_assert (fake_dev->last_called_function == dev_class->resume);
|
|
||||||
fake_dev->last_called_function = NULL;
|
|
||||||
|
|
||||||
|
|
||||||
/* The "verify" operation is still ongoing, finish it. */
|
|
||||||
orig_verify (device);
|
|
||||||
while (g_main_context_iteration (NULL, FALSE))
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
test_driver_current_action (void)
|
test_driver_current_action (void)
|
||||||
{
|
{
|
||||||
|
@ -2772,32 +2350,32 @@ test_driver_action_get_cancellable_open (void)
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
test_driver_action_get_cancellable_open_internal_vfunc (FpDevice *device)
|
test_driver_action_get_cancellable_open_fail_vfunc (FpDevice *device)
|
||||||
{
|
{
|
||||||
FpiDeviceFake *fake_dev = FPI_DEVICE_FAKE (device);
|
FpiDeviceFake *fake_dev = FPI_DEVICE_FAKE (device);
|
||||||
|
|
||||||
g_assert_cmpuint (fpi_device_get_current_action (device), ==, FPI_DEVICE_ACTION_OPEN);
|
g_assert_cmpuint (fpi_device_get_current_action (device), ==, FPI_DEVICE_ACTION_OPEN);
|
||||||
fake_dev->last_called_function = test_driver_action_get_cancellable_open_internal_vfunc;
|
fake_dev->last_called_function = test_driver_action_get_cancellable_open_fail_vfunc;
|
||||||
|
|
||||||
g_assert_true (G_IS_CANCELLABLE (fpi_device_get_cancellable (device)));
|
g_assert_false (G_IS_CANCELLABLE (fpi_device_get_cancellable (device)));
|
||||||
|
|
||||||
fpi_device_open_complete (device, NULL);
|
fpi_device_open_complete (device, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
test_driver_action_get_cancellable_open_internal (void)
|
test_driver_action_get_cancellable_open_fail (void)
|
||||||
{
|
{
|
||||||
g_autoptr(FpAutoResetClass) dev_class = auto_reset_device_class ();
|
g_autoptr(FpAutoResetClass) dev_class = auto_reset_device_class ();
|
||||||
g_autoptr(FpAutoCloseDevice) device = NULL;
|
g_autoptr(FpAutoCloseDevice) device = NULL;
|
||||||
FpiDeviceFake *fake_dev;
|
FpiDeviceFake *fake_dev;
|
||||||
|
|
||||||
dev_class->open = test_driver_action_get_cancellable_open_internal_vfunc;
|
dev_class->open = test_driver_action_get_cancellable_open_fail_vfunc;
|
||||||
device = g_object_new (FPI_TYPE_DEVICE_FAKE, NULL);
|
device = g_object_new (FPI_TYPE_DEVICE_FAKE, NULL);
|
||||||
fake_dev = FPI_DEVICE_FAKE (device);
|
fake_dev = FPI_DEVICE_FAKE (device);
|
||||||
|
|
||||||
g_assert_true (fp_device_open_sync (device, NULL, NULL));
|
g_assert_true (fp_device_open_sync (device, NULL, NULL));
|
||||||
|
|
||||||
g_assert (fake_dev->last_called_function == test_driver_action_get_cancellable_open_internal_vfunc);
|
g_assert (fake_dev->last_called_function == test_driver_action_get_cancellable_open_fail_vfunc);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -2822,11 +2400,7 @@ test_driver_action_is_cancelled_open_vfunc (FpDevice *device)
|
||||||
g_assert_true (G_IS_CANCELLABLE (fpi_device_get_cancellable (device)));
|
g_assert_true (G_IS_CANCELLABLE (fpi_device_get_cancellable (device)));
|
||||||
g_assert_false (fpi_device_action_is_cancelled (device));
|
g_assert_false (fpi_device_action_is_cancelled (device));
|
||||||
|
|
||||||
if (fake_dev->ext_cancellable)
|
|
||||||
g_cancellable_cancel (fake_dev->ext_cancellable);
|
|
||||||
else
|
|
||||||
g_cancellable_cancel (fpi_device_get_cancellable (device));
|
g_cancellable_cancel (fpi_device_get_cancellable (device));
|
||||||
|
|
||||||
g_assert_true (fpi_device_action_is_cancelled (device));
|
g_assert_true (fpi_device_action_is_cancelled (device));
|
||||||
|
|
||||||
fpi_device_open_complete (device, NULL);
|
fpi_device_open_complete (device, NULL);
|
||||||
|
@ -2845,34 +2419,13 @@ test_driver_action_is_cancelled_open (void)
|
||||||
device = g_object_new (FPI_TYPE_DEVICE_FAKE, NULL);
|
device = g_object_new (FPI_TYPE_DEVICE_FAKE, NULL);
|
||||||
fake_dev = FPI_DEVICE_FAKE (device);
|
fake_dev = FPI_DEVICE_FAKE (device);
|
||||||
|
|
||||||
cancellable = fake_dev->ext_cancellable = g_cancellable_new ();
|
cancellable = g_cancellable_new ();
|
||||||
g_assert_false (fp_device_open_sync (device, cancellable, &error));
|
g_assert_false (fp_device_open_sync (device, cancellable, &error));
|
||||||
g_assert_error (error, G_IO_ERROR, G_IO_ERROR_CANCELLED);
|
g_assert_error (error, G_IO_ERROR, G_IO_ERROR_CANCELLED);
|
||||||
|
|
||||||
g_assert (fake_dev->last_called_function == test_driver_action_is_cancelled_open_vfunc);
|
g_assert (fake_dev->last_called_function == test_driver_action_is_cancelled_open_vfunc);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
test_driver_action_internally_cancelled_open (void)
|
|
||||||
{
|
|
||||||
g_autoptr(FpAutoResetClass) dev_class = auto_reset_device_class ();
|
|
||||||
g_autoptr(FpAutoCloseDevice) device = NULL;
|
|
||||||
g_autoptr(GCancellable) cancellable = NULL;
|
|
||||||
g_autoptr(GError) error = NULL;
|
|
||||||
FpiDeviceFake *fake_dev;
|
|
||||||
|
|
||||||
dev_class->open = test_driver_action_is_cancelled_open_vfunc;
|
|
||||||
device = g_object_new (FPI_TYPE_DEVICE_FAKE, NULL);
|
|
||||||
fake_dev = FPI_DEVICE_FAKE (device);
|
|
||||||
|
|
||||||
/* No error, just some internal cancellation but we let nothing happen externally. */
|
|
||||||
cancellable = g_cancellable_new ();
|
|
||||||
g_assert_true (fp_device_open_sync (device, cancellable, &error));
|
|
||||||
g_assert_null (error);
|
|
||||||
|
|
||||||
g_assert (fake_dev->last_called_function == test_driver_action_is_cancelled_open_vfunc);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
test_driver_action_is_cancelled_error (void)
|
test_driver_action_is_cancelled_error (void)
|
||||||
{
|
{
|
||||||
|
@ -3321,14 +2874,6 @@ main (int argc, char *argv[])
|
||||||
g_test_add_func ("/driver/identify/not_reported", test_driver_identify_not_reported);
|
g_test_add_func ("/driver/identify/not_reported", test_driver_identify_not_reported);
|
||||||
g_test_add_func ("/driver/identify/complete_retry", test_driver_identify_complete_retry);
|
g_test_add_func ("/driver/identify/complete_retry", test_driver_identify_complete_retry);
|
||||||
g_test_add_func ("/driver/identify/report_no_cb", test_driver_identify_report_no_callback);
|
g_test_add_func ("/driver/identify/report_no_cb", test_driver_identify_report_no_callback);
|
||||||
|
|
||||||
g_test_add_func ("/driver/identify/suspend_continues", test_driver_identify_suspend_continues);
|
|
||||||
g_test_add_func ("/driver/identify/suspend_succeeds", test_driver_identify_suspend_succeeds);
|
|
||||||
g_test_add_func ("/driver/identify/suspend_busy_error", test_driver_identify_suspend_busy_error);
|
|
||||||
g_test_add_func ("/driver/identify/suspend_while_idle", test_driver_identify_suspend_while_idle);
|
|
||||||
|
|
||||||
g_test_add_func ("/driver/identify/warmup_cooldown", test_driver_identify_warmup_cooldown);
|
|
||||||
|
|
||||||
g_test_add_func ("/driver/capture", test_driver_capture);
|
g_test_add_func ("/driver/capture", test_driver_capture);
|
||||||
g_test_add_func ("/driver/capture/not_supported", test_driver_capture_not_supported);
|
g_test_add_func ("/driver/capture/not_supported", test_driver_capture_not_supported);
|
||||||
g_test_add_func ("/driver/capture/error", test_driver_capture_error);
|
g_test_add_func ("/driver/capture/error", test_driver_capture_error);
|
||||||
|
@ -3342,15 +2887,12 @@ main (int argc, char *argv[])
|
||||||
g_test_add_func ("/driver/cancel", test_driver_cancel);
|
g_test_add_func ("/driver/cancel", test_driver_cancel);
|
||||||
g_test_add_func ("/driver/cancel/fail", test_driver_cancel_fail);
|
g_test_add_func ("/driver/cancel/fail", test_driver_cancel_fail);
|
||||||
|
|
||||||
g_test_add_func ("/driver/critical", test_driver_critical);
|
|
||||||
|
|
||||||
g_test_add_func ("/driver/get_current_action", test_driver_current_action);
|
g_test_add_func ("/driver/get_current_action", test_driver_current_action);
|
||||||
g_test_add_func ("/driver/get_current_action/open", test_driver_current_action_open);
|
g_test_add_func ("/driver/get_current_action/open", test_driver_current_action_open);
|
||||||
g_test_add_func ("/driver/get_cancellable/error", test_driver_action_get_cancellable_error);
|
g_test_add_func ("/driver/get_cancellable/error", test_driver_action_get_cancellable_error);
|
||||||
g_test_add_func ("/driver/get_cancellable/open", test_driver_action_get_cancellable_open);
|
g_test_add_func ("/driver/get_cancellable/open", test_driver_action_get_cancellable_open);
|
||||||
g_test_add_func ("/driver/get_cancellable/open/internal", test_driver_action_get_cancellable_open_internal);
|
g_test_add_func ("/driver/get_cancellable/open/fail", test_driver_action_get_cancellable_open_fail);
|
||||||
g_test_add_func ("/driver/action_is_cancelled/open", test_driver_action_is_cancelled_open);
|
g_test_add_func ("/driver/action_is_cancelled/open", test_driver_action_is_cancelled_open);
|
||||||
g_test_add_func ("/driver/action_is_cancelled/open/internal", test_driver_action_internally_cancelled_open);
|
|
||||||
g_test_add_func ("/driver/action_is_cancelled/error", test_driver_action_is_cancelled_error);
|
g_test_add_func ("/driver/action_is_cancelled/error", test_driver_action_is_cancelled_error);
|
||||||
g_test_add_func ("/driver/complete_action/all/error", test_driver_complete_actions_errors);
|
g_test_add_func ("/driver/complete_action/all/error", test_driver_complete_actions_errors);
|
||||||
g_test_add_func ("/driver/action_error/error", test_driver_action_error_error);
|
g_test_add_func ("/driver/action_error/error", test_driver_action_error_error);
|
||||||
|
|
|
@ -18,7 +18,7 @@ try:
|
||||||
if version < (0, 13, 2):
|
if version < (0, 13, 2):
|
||||||
print('umockdev is too old for test to be reliable, expect random failures!')
|
print('umockdev is too old for test to be reliable, expect random failures!')
|
||||||
print('Please update umockdev to at least 0.13.2.')
|
print('Please update umockdev to at least 0.13.2.')
|
||||||
pcap_supported = version >= (0, 16, 3) or os.getenv('CI_PROJECT_NAME') == "libfprint"
|
pcap_supported = version >= (0, 16) or os.getenv('CI_PROJECT_NAME') == "libfprint"
|
||||||
spi_supported = version >= (0, 16) or os.getenv('CI_PROJECT_NAME') == "libfprint"
|
spi_supported = version >= (0, 16) or os.getenv('CI_PROJECT_NAME') == "libfprint"
|
||||||
|
|
||||||
except FileNotFoundError:
|
except FileNotFoundError:
|
||||||
|
|
Binary file not shown.
Binary file not shown.
Before Width: | Height: | Size: 27 KiB |
|
@ -1,319 +0,0 @@
|
||||||
P: /devices/pci0000:00/0000:00:1a.0/usb1/1-1/1-1.3
|
|
||||||
N: bus/usb/001/003=12010101000000087E14162002000102000109022700010100A0320904000003FF000000070581024000000705020240000007058303040014
|
|
||||||
E: DEVNAME=/dev/bus/usb/001/003
|
|
||||||
E: DEVTYPE=usb_device
|
|
||||||
E: DRIVER=usb
|
|
||||||
E: PRODUCT=147e/2016/2
|
|
||||||
E: TYPE=0/0/0
|
|
||||||
E: BUSNUM=001
|
|
||||||
E: DEVNUM=003
|
|
||||||
E: MAJOR=189
|
|
||||||
E: MINOR=2
|
|
||||||
E: SUBSYSTEM=usb
|
|
||||||
E: ID_VENDOR=UPEK
|
|
||||||
E: ID_VENDOR_ENC=UPEK
|
|
||||||
E: ID_VENDOR_ID=147e
|
|
||||||
E: ID_MODEL=Biometric_Coprocessor
|
|
||||||
E: ID_MODEL_ENC=Biometric\x20Coprocessor
|
|
||||||
E: ID_MODEL_ID=2016
|
|
||||||
E: ID_REVISION=0002
|
|
||||||
E: ID_SERIAL=UPEK_Biometric_Coprocessor
|
|
||||||
E: ID_BUS=usb
|
|
||||||
E: ID_USB_INTERFACES=:ff0000:
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Upek
|
|
||||||
E: ID_MODEL_FROM_DATABASE=Biometric Touchchip/Touchstrip Fingerprint Sensor
|
|
||||||
E: ID_PATH=pci-0000:00:1a.0-usb-0:1.3
|
|
||||||
E: ID_PATH_TAG=pci-0000_00_1a_0-usb-0_1_3
|
|
||||||
E: LIBFPRINT_DRIVER=Upek TouchChip Fingerprint Coprocessor
|
|
||||||
E: ID_FOR_SEAT=usb-pci-0000_00_1a_0-usb-0_1_3
|
|
||||||
E: TAGS=:seat:
|
|
||||||
E: CURRENT_TAGS=:seat:
|
|
||||||
A: authorized=1\n
|
|
||||||
A: avoid_reset_quirk=0\n
|
|
||||||
A: bConfigurationValue=1\n
|
|
||||||
A: bDeviceClass=00\n
|
|
||||||
A: bDeviceProtocol=00\n
|
|
||||||
A: bDeviceSubClass=00\n
|
|
||||||
A: bMaxPacketSize0=8\n
|
|
||||||
A: bMaxPower=100mA\n
|
|
||||||
A: bNumConfigurations=1\n
|
|
||||||
A: bNumInterfaces= 1\n
|
|
||||||
A: bcdDevice=0002\n
|
|
||||||
A: bmAttributes=a0\n
|
|
||||||
A: busnum=1\n
|
|
||||||
A: configuration=
|
|
||||||
H: descriptors=12010101000000087E14162002000102000109022700010100A0320904000003FF000000070581024000000705020240000007058303040014
|
|
||||||
A: dev=189:2\n
|
|
||||||
A: devnum=3\n
|
|
||||||
A: devpath=1.3\n
|
|
||||||
L: driver=../../../../../../bus/usb/drivers/usb
|
|
||||||
L: firmware_node=../../../../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:2d/device:2e/device:2f/device:32
|
|
||||||
A: idProduct=2016\n
|
|
||||||
A: idVendor=147e\n
|
|
||||||
A: ltm_capable=no\n
|
|
||||||
A: manufacturer=UPEK\n
|
|
||||||
A: maxchild=0\n
|
|
||||||
L: port=../1-1:1.0/1-1-port3
|
|
||||||
A: power/active_duration=757220\n
|
|
||||||
A: power/async=enabled\n
|
|
||||||
A: power/autosuspend=2\n
|
|
||||||
A: power/autosuspend_delay_ms=2000\n
|
|
||||||
A: power/connected_duration=857556\n
|
|
||||||
A: power/control=auto\n
|
|
||||||
A: power/level=auto\n
|
|
||||||
A: power/persist=1\n
|
|
||||||
A: power/runtime_active_kids=0\n
|
|
||||||
A: power/runtime_active_time=762579\n
|
|
||||||
A: power/runtime_enabled=enabled\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=94791\n
|
|
||||||
A: power/runtime_usage=0\n
|
|
||||||
A: power/wakeup=disabled\n
|
|
||||||
A: power/wakeup_abort_count=\n
|
|
||||||
A: power/wakeup_active=\n
|
|
||||||
A: power/wakeup_active_count=\n
|
|
||||||
A: power/wakeup_count=\n
|
|
||||||
A: power/wakeup_expire_count=\n
|
|
||||||
A: power/wakeup_last_time_ms=\n
|
|
||||||
A: power/wakeup_max_time_ms=\n
|
|
||||||
A: power/wakeup_total_time_ms=\n
|
|
||||||
A: product=Biometric Coprocessor\n
|
|
||||||
A: quirks=0x0\n
|
|
||||||
A: removable=fixed\n
|
|
||||||
A: rx_lanes=1\n
|
|
||||||
A: speed=12\n
|
|
||||||
A: tx_lanes=1\n
|
|
||||||
A: urbnum=620\n
|
|
||||||
A: version= 1.01\n
|
|
||||||
|
|
||||||
P: /devices/pci0000:00/0000:00:1a.0/usb1/1-1
|
|
||||||
N: bus/usb/001/002=12010002090001408780200000000000000109021900010100E0000904000001090000000705810301000C
|
|
||||||
E: DEVNAME=/dev/bus/usb/001/002
|
|
||||||
E: DEVTYPE=usb_device
|
|
||||||
E: DRIVER=usb
|
|
||||||
E: PRODUCT=8087/20/0
|
|
||||||
E: TYPE=9/0/1
|
|
||||||
E: BUSNUM=001
|
|
||||||
E: DEVNUM=002
|
|
||||||
E: MAJOR=189
|
|
||||||
E: MINOR=1
|
|
||||||
E: SUBSYSTEM=usb
|
|
||||||
E: ID_VENDOR=8087
|
|
||||||
E: ID_VENDOR_ENC=8087
|
|
||||||
E: ID_VENDOR_ID=8087
|
|
||||||
E: ID_MODEL=0020
|
|
||||||
E: ID_MODEL_ENC=0020
|
|
||||||
E: ID_MODEL_ID=0020
|
|
||||||
E: ID_REVISION=0000
|
|
||||||
E: ID_SERIAL=8087_0020
|
|
||||||
E: ID_BUS=usb
|
|
||||||
E: ID_USB_INTERFACES=:090000:
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Intel Corp.
|
|
||||||
E: ID_MODEL_FROM_DATABASE=Integrated Rate Matching Hub
|
|
||||||
E: ID_PATH=pci-0000:00:1a.0-usb-0:1
|
|
||||||
E: ID_PATH_TAG=pci-0000_00_1a_0-usb-0_1
|
|
||||||
E: ID_FOR_SEAT=usb-pci-0000_00_1a_0-usb-0_1
|
|
||||||
E: TAGS=:seat:
|
|
||||||
E: CURRENT_TAGS=:seat:
|
|
||||||
A: authorized=1\n
|
|
||||||
A: avoid_reset_quirk=0\n
|
|
||||||
A: bConfigurationValue=1\n
|
|
||||||
A: bDeviceClass=09\n
|
|
||||||
A: bDeviceProtocol=01\n
|
|
||||||
A: bDeviceSubClass=00\n
|
|
||||||
A: bMaxPacketSize0=64\n
|
|
||||||
A: bMaxPower=0mA\n
|
|
||||||
A: bNumConfigurations=1\n
|
|
||||||
A: bNumInterfaces= 1\n
|
|
||||||
A: bcdDevice=0000\n
|
|
||||||
A: bmAttributes=e0\n
|
|
||||||
A: busnum=1\n
|
|
||||||
A: configuration=
|
|
||||||
H: descriptors=12010002090001408780200000000000000109021900010100E0000904000001090000000705810301000C
|
|
||||||
A: dev=189:1\n
|
|
||||||
A: devnum=2\n
|
|
||||||
A: devpath=1\n
|
|
||||||
L: driver=../../../../../bus/usb/drivers/usb
|
|
||||||
L: firmware_node=../../../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:2d/device:2e/device:2f
|
|
||||||
A: idProduct=0020\n
|
|
||||||
A: idVendor=8087\n
|
|
||||||
A: ltm_capable=no\n
|
|
||||||
A: maxchild=6\n
|
|
||||||
L: port=../1-0:1.0/usb1-port1
|
|
||||||
A: power/active_duration=776368\n
|
|
||||||
A: power/async=enabled\n
|
|
||||||
A: power/autosuspend=0\n
|
|
||||||
A: power/autosuspend_delay_ms=0\n
|
|
||||||
A: power/connected_duration=858060\n
|
|
||||||
A: power/control=auto\n
|
|
||||||
A: power/level=auto\n
|
|
||||||
A: power/runtime_active_kids=3\n
|
|
||||||
A: power/runtime_active_time=779955\n
|
|
||||||
A: power/runtime_enabled=enabled\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=77820\n
|
|
||||||
A: power/runtime_usage=0\n
|
|
||||||
A: power/wakeup=disabled\n
|
|
||||||
A: power/wakeup_abort_count=\n
|
|
||||||
A: power/wakeup_active=\n
|
|
||||||
A: power/wakeup_active_count=\n
|
|
||||||
A: power/wakeup_count=\n
|
|
||||||
A: power/wakeup_expire_count=\n
|
|
||||||
A: power/wakeup_last_time_ms=\n
|
|
||||||
A: power/wakeup_max_time_ms=\n
|
|
||||||
A: power/wakeup_total_time_ms=\n
|
|
||||||
A: quirks=0x0\n
|
|
||||||
A: removable=fixed\n
|
|
||||||
A: rx_lanes=1\n
|
|
||||||
A: speed=480\n
|
|
||||||
A: tx_lanes=1\n
|
|
||||||
A: urbnum=2620\n
|
|
||||||
A: version= 2.00\n
|
|
||||||
|
|
||||||
P: /devices/pci0000:00/0000:00:1a.0/usb1
|
|
||||||
N: bus/usb/001/001=12010002090000406B1D020010050302010109021900010100E0000904000001090000000705810304000C
|
|
||||||
E: DEVNAME=/dev/bus/usb/001/001
|
|
||||||
E: DEVTYPE=usb_device
|
|
||||||
E: DRIVER=usb
|
|
||||||
E: PRODUCT=1d6b/2/510
|
|
||||||
E: TYPE=9/0/0
|
|
||||||
E: BUSNUM=001
|
|
||||||
E: DEVNUM=001
|
|
||||||
E: MAJOR=189
|
|
||||||
E: MINOR=0
|
|
||||||
E: SUBSYSTEM=usb
|
|
||||||
E: ID_VENDOR=Linux_5.10.0-8-amd64_ehci_hcd
|
|
||||||
E: ID_VENDOR_ENC=Linux\x205.10.0-8-amd64\x20ehci_hcd
|
|
||||||
E: ID_VENDOR_ID=1d6b
|
|
||||||
E: ID_MODEL=EHCI_Host_Controller
|
|
||||||
E: ID_MODEL_ENC=EHCI\x20Host\x20Controller
|
|
||||||
E: ID_MODEL_ID=0002
|
|
||||||
E: ID_REVISION=0510
|
|
||||||
E: ID_SERIAL=Linux_5.10.0-8-amd64_ehci_hcd_EHCI_Host_Controller_0000:00:1a.0
|
|
||||||
E: ID_SERIAL_SHORT=0000:00:1a.0
|
|
||||||
E: ID_BUS=usb
|
|
||||||
E: ID_USB_INTERFACES=:090000:
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Linux Foundation
|
|
||||||
E: ID_AUTOSUSPEND=1
|
|
||||||
E: ID_MODEL_FROM_DATABASE=2.0 root hub
|
|
||||||
E: ID_PATH=pci-0000:00:1a.0
|
|
||||||
E: ID_PATH_TAG=pci-0000_00_1a_0
|
|
||||||
E: ID_FOR_SEAT=usb-pci-0000_00_1a_0
|
|
||||||
E: TAGS=:seat:
|
|
||||||
E: CURRENT_TAGS=:seat:
|
|
||||||
A: authorized=1\n
|
|
||||||
A: authorized_default=1\n
|
|
||||||
A: avoid_reset_quirk=0\n
|
|
||||||
A: bConfigurationValue=1\n
|
|
||||||
A: bDeviceClass=09\n
|
|
||||||
A: bDeviceProtocol=00\n
|
|
||||||
A: bDeviceSubClass=00\n
|
|
||||||
A: bMaxPacketSize0=64\n
|
|
||||||
A: bMaxPower=0mA\n
|
|
||||||
A: bNumConfigurations=1\n
|
|
||||||
A: bNumInterfaces= 1\n
|
|
||||||
A: bcdDevice=0510\n
|
|
||||||
A: bmAttributes=e0\n
|
|
||||||
A: busnum=1\n
|
|
||||||
A: configuration=
|
|
||||||
H: descriptors=12010002090000406B1D020010050302010109021900010100E0000904000001090000000705810304000C
|
|
||||||
A: dev=189:0\n
|
|
||||||
A: devnum=1\n
|
|
||||||
A: devpath=0\n
|
|
||||||
L: driver=../../../../bus/usb/drivers/usb
|
|
||||||
L: firmware_node=../../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:2d/device:2e
|
|
||||||
A: idProduct=0002\n
|
|
||||||
A: idVendor=1d6b\n
|
|
||||||
A: interface_authorized_default=1\n
|
|
||||||
A: ltm_capable=no\n
|
|
||||||
A: manufacturer=Linux 5.10.0-8-amd64 ehci_hcd\n
|
|
||||||
A: maxchild=3\n
|
|
||||||
A: power/active_duration=780512\n
|
|
||||||
A: power/async=enabled\n
|
|
||||||
A: power/autosuspend=0\n
|
|
||||||
A: power/autosuspend_delay_ms=0\n
|
|
||||||
A: power/connected_duration=858228\n
|
|
||||||
A: power/control=auto\n
|
|
||||||
A: power/level=auto\n
|
|
||||||
A: power/runtime_active_kids=1\n
|
|
||||||
A: power/runtime_active_time=780500\n
|
|
||||||
A: power/runtime_enabled=enabled\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=77697\n
|
|
||||||
A: power/runtime_usage=0\n
|
|
||||||
A: power/wakeup=disabled\n
|
|
||||||
A: power/wakeup_abort_count=\n
|
|
||||||
A: power/wakeup_active=\n
|
|
||||||
A: power/wakeup_active_count=\n
|
|
||||||
A: power/wakeup_count=\n
|
|
||||||
A: power/wakeup_expire_count=\n
|
|
||||||
A: power/wakeup_last_time_ms=\n
|
|
||||||
A: power/wakeup_max_time_ms=\n
|
|
||||||
A: power/wakeup_total_time_ms=\n
|
|
||||||
A: product=EHCI Host Controller\n
|
|
||||||
A: quirks=0x0\n
|
|
||||||
A: removable=unknown\n
|
|
||||||
A: rx_lanes=1\n
|
|
||||||
A: serial=0000:00:1a.0\n
|
|
||||||
A: speed=480\n
|
|
||||||
A: tx_lanes=1\n
|
|
||||||
A: urbnum=1071\n
|
|
||||||
A: version= 2.00\n
|
|
||||||
|
|
||||||
P: /devices/pci0000:00/0000:00:1a.0
|
|
||||||
E: DRIVER=ehci-pci
|
|
||||||
E: PCI_CLASS=C0320
|
|
||||||
E: PCI_ID=8086:3B3C
|
|
||||||
E: PCI_SUBSYS_ID=17AA:2163
|
|
||||||
E: PCI_SLOT_NAME=0000:00:1a.0
|
|
||||||
E: MODALIAS=pci:v00008086d00003B3Csv000017AAsd00002163bc0Csc03i20
|
|
||||||
E: SUBSYSTEM=pci
|
|
||||||
E: ID_PCI_CLASS_FROM_DATABASE=Serial bus controller
|
|
||||||
E: ID_PCI_SUBCLASS_FROM_DATABASE=USB controller
|
|
||||||
E: ID_PCI_INTERFACE_FROM_DATABASE=EHCI
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Intel Corporation
|
|
||||||
E: ID_MODEL_FROM_DATABASE=5 Series/3400 Series Chipset USB2 Enhanced Host Controller
|
|
||||||
A: ari_enabled=0\n
|
|
||||||
A: broken_parity_status=0\n
|
|
||||||
A: class=0x0c0320\n
|
|
||||||
A: companion=
|
|
||||||
H: config=86803C3B060190020620030C00000000008072F2000000000000000000000000000000000000000000000000AA1763210000000050000000000000000B040000000000000000000000000000000000000158C2C9000000000A98A020000000002020A7070000000001000001000008C00000DF3F0000000000000000000000000000800011890C13A000000000000000000000000000000013000603000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000AAFF00000000000000000000000000000000000000000004F340BB0000000088858000870F060828171B30
|
|
||||||
A: consistent_dma_mask_bits=32\n
|
|
||||||
A: d3cold_allowed=1\n
|
|
||||||
A: device=0x3b3c\n
|
|
||||||
A: dma_mask_bits=32\n
|
|
||||||
L: driver=../../../bus/pci/drivers/ehci-pci
|
|
||||||
A: driver_override=(null)\n
|
|
||||||
A: enable=1\n
|
|
||||||
L: firmware_node=../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:2d
|
|
||||||
A: irq=23\n
|
|
||||||
A: local_cpulist=0-3\n
|
|
||||||
A: local_cpus=f\n
|
|
||||||
A: modalias=pci:v00008086d00003B3Csv000017AAsd00002163bc0Csc03i20\n
|
|
||||||
A: msi_bus=1\n
|
|
||||||
A: numa_node=-1\n
|
|
||||||
A: pools=poolinfo - 0.1\nehci_sitd 0 0 96 0\nehci_itd 0 0 192 0\nehci_qh 9 42 96 1\nehci_qtd 13 42 96 1\nbuffer-2048 0 0 2048 0\nbuffer-512 0 0 512 0\nbuffer-128 0 0 128 0\nbuffer-32 0 0 32 0\n
|
|
||||||
A: power/async=enabled\n
|
|
||||||
A: power/control=on\n
|
|
||||||
A: power/runtime_active_kids=1\n
|
|
||||||
A: power/runtime_active_time=859876\n
|
|
||||||
A: power/runtime_enabled=forbidden\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=0\n
|
|
||||||
A: power/runtime_usage=1\n
|
|
||||||
A: power/wakeup=enabled\n
|
|
||||||
A: power/wakeup_abort_count=0\n
|
|
||||||
A: power/wakeup_active=0\n
|
|
||||||
A: power/wakeup_active_count=0\n
|
|
||||||
A: power/wakeup_count=0\n
|
|
||||||
A: power/wakeup_expire_count=0\n
|
|
||||||
A: power/wakeup_last_time_ms=0\n
|
|
||||||
A: power/wakeup_max_time_ms=0\n
|
|
||||||
A: power/wakeup_total_time_ms=0\n
|
|
||||||
A: resource=0x00000000f2728000 0x00000000f27283ff 0x0000000000040200\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n
|
|
||||||
A: revision=0x06\n
|
|
||||||
A: subsystem_device=0x2163\n
|
|
||||||
A: subsystem_vendor=0x17aa\n
|
|
||||||
A: uframe_periodic_max=100\n
|
|
||||||
A: vendor=0x8086\n
|
|
||||||
|
|
Binary file not shown.
Binary file not shown.
Before Width: | Height: | Size: 94 KiB |
|
@ -1,220 +0,0 @@
|
||||||
P: /devices/pci0000:00/0000:00:14.0/usb1/1-10
|
|
||||||
N: bus/usb/001/050=1201000200000040BA050A000301010203010902200001010080640904000002FFFFFF000705810340000807058202400000
|
|
||||||
E: DEVNAME=/dev/bus/usb/001/050
|
|
||||||
E: DEVTYPE=usb_device
|
|
||||||
E: DRIVER=usb
|
|
||||||
E: PRODUCT=5ba/a/103
|
|
||||||
E: TYPE=0/0/0
|
|
||||||
E: BUSNUM=001
|
|
||||||
E: DEVNUM=050
|
|
||||||
E: MAJOR=189
|
|
||||||
E: MINOR=49
|
|
||||||
E: SUBSYSTEM=usb
|
|
||||||
E: ID_VENDOR=DigitalPersona__Inc.
|
|
||||||
E: ID_VENDOR_ENC=DigitalPersona\x2c\x20Inc.
|
|
||||||
E: ID_VENDOR_ID=05ba
|
|
||||||
E: ID_MODEL=U.are.U®_4500_Fingerprint_Reader
|
|
||||||
E: ID_MODEL_ENC=U.are.U®\x204500\x20Fingerprint\x20Reader
|
|
||||||
E: ID_MODEL_ID=000a
|
|
||||||
E: ID_REVISION=0103
|
|
||||||
E: ID_SERIAL=DigitalPersona__Inc._U.are.U®_4500_Fingerprint_Reader__FB0B9071-2E08-7742-BC16-2FAA247CEF66_
|
|
||||||
E: ID_SERIAL_SHORT=_FB0B9071-2E08-7742-BC16-2FAA247CEF66_
|
|
||||||
E: ID_BUS=usb
|
|
||||||
E: ID_USB_INTERFACES=:ffffff:
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=DigitalPersona, Inc.
|
|
||||||
E: ID_AUTOSUSPEND=1
|
|
||||||
E: ID_MODEL_FROM_DATABASE=Fingerprint Reader
|
|
||||||
E: ID_PATH=pci-0000:00:14.0-usb-0:10
|
|
||||||
E: ID_PATH_TAG=pci-0000_00_14_0-usb-0_10
|
|
||||||
E: LIBFPRINT_DRIVER=Digital Persona U.are.U 4000/4000B/4500
|
|
||||||
E: ID_FOR_SEAT=usb-pci-0000_00_14_0-usb-0_10
|
|
||||||
E: TAGS=:seat:
|
|
||||||
E: CURRENT_TAGS=:seat:
|
|
||||||
A: authorized=1\n
|
|
||||||
A: avoid_reset_quirk=0\n
|
|
||||||
A: bConfigurationValue=1\n
|
|
||||||
A: bDeviceClass=00\n
|
|
||||||
A: bDeviceProtocol=00\n
|
|
||||||
A: bDeviceSubClass=00\n
|
|
||||||
A: bMaxPacketSize0=64\n
|
|
||||||
A: bMaxPower=200mA\n
|
|
||||||
A: bNumConfigurations=1\n
|
|
||||||
A: bNumInterfaces= 1\n
|
|
||||||
A: bcdDevice=0103\n
|
|
||||||
A: bmAttributes=80\n
|
|
||||||
A: busnum=1\n
|
|
||||||
A: configuration=
|
|
||||||
H: descriptors=1201000200000040BA050A000301010203010902200001010080640904000002FFFFFF000705810340000807058202400000
|
|
||||||
A: dev=189:49\n
|
|
||||||
A: devnum=50\n
|
|
||||||
A: devpath=10\n
|
|
||||||
L: driver=../../../../../bus/usb/drivers/usb
|
|
||||||
L: firmware_node=../../../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:4c/device:4d/device:57
|
|
||||||
A: idProduct=000a\n
|
|
||||||
A: idVendor=05ba\n
|
|
||||||
A: ltm_capable=no\n
|
|
||||||
A: manufacturer=DigitalPersona, Inc.\n
|
|
||||||
A: maxchild=0\n
|
|
||||||
L: port=../1-0:1.0/usb1-port10
|
|
||||||
A: power/active_duration=2761\n
|
|
||||||
A: power/autosuspend=2\n
|
|
||||||
A: power/autosuspend_delay_ms=2000\n
|
|
||||||
A: power/connected_duration=118841\n
|
|
||||||
A: power/control=auto\n
|
|
||||||
A: power/level=auto\n
|
|
||||||
A: power/persist=0\n
|
|
||||||
A: power/runtime_active_time=2616\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=115982\n
|
|
||||||
A: product=U.are.U\302\256 4500 Fingerprint Reader\n
|
|
||||||
A: quirks=0x0\n
|
|
||||||
A: removable=removable\n
|
|
||||||
A: rx_lanes=1\n
|
|
||||||
A: serial={FB0B9071-2E08-7742-BC16-2FAA247CEF66}\n
|
|
||||||
A: speed=12\n
|
|
||||||
A: tx_lanes=1\n
|
|
||||||
A: urbnum=13\n
|
|
||||||
A: version= 2.00\n
|
|
||||||
|
|
||||||
P: /devices/pci0000:00/0000:00:14.0/usb1
|
|
||||||
N: bus/usb/001/001=12010002090001406B1D020013050302010109021900010100E0000904000001090000000705810304000C
|
|
||||||
E: DEVNAME=/dev/bus/usb/001/001
|
|
||||||
E: DEVTYPE=usb_device
|
|
||||||
E: DRIVER=usb
|
|
||||||
E: PRODUCT=1d6b/2/513
|
|
||||||
E: TYPE=9/0/1
|
|
||||||
E: BUSNUM=001
|
|
||||||
E: DEVNUM=001
|
|
||||||
E: MAJOR=189
|
|
||||||
E: MINOR=0
|
|
||||||
E: SUBSYSTEM=usb
|
|
||||||
E: ID_VENDOR=Linux_5.13.12-200.fc34.x86_64_xhci-hcd
|
|
||||||
E: ID_VENDOR_ENC=Linux\x205.13.12-200.fc34.x86_64\x20xhci-hcd
|
|
||||||
E: ID_VENDOR_ID=1d6b
|
|
||||||
E: ID_MODEL=xHCI_Host_Controller
|
|
||||||
E: ID_MODEL_ENC=xHCI\x20Host\x20Controller
|
|
||||||
E: ID_MODEL_ID=0002
|
|
||||||
E: ID_REVISION=0513
|
|
||||||
E: ID_SERIAL=Linux_5.13.12-200.fc34.x86_64_xhci-hcd_xHCI_Host_Controller_0000:00:14.0
|
|
||||||
E: ID_SERIAL_SHORT=0000:00:14.0
|
|
||||||
E: ID_BUS=usb
|
|
||||||
E: ID_USB_INTERFACES=:090000:
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Linux Foundation
|
|
||||||
E: ID_AUTOSUSPEND=1
|
|
||||||
E: ID_MODEL_FROM_DATABASE=2.0 root hub
|
|
||||||
E: ID_PATH=pci-0000:00:14.0
|
|
||||||
E: ID_PATH_TAG=pci-0000_00_14_0
|
|
||||||
E: ID_FOR_SEAT=usb-pci-0000_00_14_0
|
|
||||||
E: TAGS=:seat:
|
|
||||||
E: CURRENT_TAGS=:seat:
|
|
||||||
A: authorized=1\n
|
|
||||||
A: authorized_default=1\n
|
|
||||||
A: avoid_reset_quirk=0\n
|
|
||||||
A: bConfigurationValue=1\n
|
|
||||||
A: bDeviceClass=09\n
|
|
||||||
A: bDeviceProtocol=01\n
|
|
||||||
A: bDeviceSubClass=00\n
|
|
||||||
A: bMaxPacketSize0=64\n
|
|
||||||
A: bMaxPower=0mA\n
|
|
||||||
A: bNumConfigurations=1\n
|
|
||||||
A: bNumInterfaces= 1\n
|
|
||||||
A: bcdDevice=0513\n
|
|
||||||
A: bmAttributes=e0\n
|
|
||||||
A: busnum=1\n
|
|
||||||
A: configuration=
|
|
||||||
H: descriptors=12010002090001406B1D020013050302010109021900010100E0000904000001090000000705810304000C
|
|
||||||
A: dev=189:0\n
|
|
||||||
A: devnum=1\n
|
|
||||||
A: devpath=0\n
|
|
||||||
L: driver=../../../../bus/usb/drivers/usb
|
|
||||||
L: firmware_node=../../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:4c/device:4d
|
|
||||||
A: idProduct=0002\n
|
|
||||||
A: idVendor=1d6b\n
|
|
||||||
A: interface_authorized_default=1\n
|
|
||||||
A: ltm_capable=no\n
|
|
||||||
A: manufacturer=Linux 5.13.12-200.fc34.x86_64 xhci-hcd\n
|
|
||||||
A: maxchild=16\n
|
|
||||||
A: power/active_duration=837797629\n
|
|
||||||
A: power/autosuspend=0\n
|
|
||||||
A: power/autosuspend_delay_ms=0\n
|
|
||||||
A: power/connected_duration=837797629\n
|
|
||||||
A: power/control=auto\n
|
|
||||||
A: power/level=auto\n
|
|
||||||
A: power/runtime_active_time=837797626\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=0\n
|
|
||||||
A: power/wakeup=disabled\n
|
|
||||||
A: power/wakeup_abort_count=\n
|
|
||||||
A: power/wakeup_active=\n
|
|
||||||
A: power/wakeup_active_count=\n
|
|
||||||
A: power/wakeup_count=\n
|
|
||||||
A: power/wakeup_expire_count=\n
|
|
||||||
A: power/wakeup_last_time_ms=\n
|
|
||||||
A: power/wakeup_max_time_ms=\n
|
|
||||||
A: power/wakeup_total_time_ms=\n
|
|
||||||
A: product=xHCI Host Controller\n
|
|
||||||
A: quirks=0x0\n
|
|
||||||
A: removable=unknown\n
|
|
||||||
A: rx_lanes=1\n
|
|
||||||
A: serial=0000:00:14.0\n
|
|
||||||
A: speed=480\n
|
|
||||||
A: tx_lanes=1\n
|
|
||||||
A: urbnum=1498\n
|
|
||||||
A: version= 2.00\n
|
|
||||||
|
|
||||||
P: /devices/pci0000:00/0000:00:14.0
|
|
||||||
E: DRIVER=xhci_hcd
|
|
||||||
E: PCI_CLASS=C0330
|
|
||||||
E: PCI_ID=8086:A36D
|
|
||||||
E: PCI_SUBSYS_ID=17AA:312A
|
|
||||||
E: PCI_SLOT_NAME=0000:00:14.0
|
|
||||||
E: MODALIAS=pci:v00008086d0000A36Dsv000017AAsd0000312Abc0Csc03i30
|
|
||||||
E: SUBSYSTEM=pci
|
|
||||||
E: ID_PCI_CLASS_FROM_DATABASE=Serial bus controller
|
|
||||||
E: ID_PCI_SUBCLASS_FROM_DATABASE=USB controller
|
|
||||||
E: ID_PCI_INTERFACE_FROM_DATABASE=XHCI
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Intel Corporation
|
|
||||||
E: ID_MODEL_FROM_DATABASE=Cannon Lake PCH USB 3.1 xHCI Host Controller
|
|
||||||
A: ari_enabled=0\n
|
|
||||||
A: broken_parity_status=0\n
|
|
||||||
A: class=0x0c0330\n
|
|
||||||
H: config=86806DA3060490021030030C00008000040032B1000000000000000000000000000000000000000000000000AA172A31000000007000000000000000FF010000FD0134808FC6FF8300000000000000007F6DDC0F000000009A1CF40100000000316000000000000000000000000000000180C2C1080000000000000000000000059087009802E0FE0000000000000000090014F01000400100000000C10A080000080E00001800008F400200000100006000000008020000010000000400000090000000001800000005000000000000000300000C0000004000000080000000030000000000000000000000000000000000000000000000B50F110112000000
|
|
||||||
A: consistent_dma_mask_bits=64\n
|
|
||||||
A: d3cold_allowed=1\n
|
|
||||||
A: dbc=disabled\n
|
|
||||||
A: device=0xa36d\n
|
|
||||||
A: dma_mask_bits=64\n
|
|
||||||
L: driver=../../../bus/pci/drivers/xhci_hcd
|
|
||||||
A: driver_override=(null)\n
|
|
||||||
A: enable=1\n
|
|
||||||
L: firmware_node=../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:4c
|
|
||||||
A: index=3\n
|
|
||||||
A: irq=125\n
|
|
||||||
A: label=Onboard - Other\n
|
|
||||||
A: local_cpulist=0-5\n
|
|
||||||
A: local_cpus=3f\n
|
|
||||||
A: modalias=pci:v00008086d0000A36Dsv000017AAsd0000312Abc0Csc03i30\n
|
|
||||||
A: msi_bus=1\n
|
|
||||||
A: msi_irqs/125=msi\n
|
|
||||||
A: numa_node=-1\n
|
|
||||||
A: pools=poolinfo - 0.1\nbuffer-2048 0 0 2048 0\nbuffer-512 0 0 512 0\nbuffer-128 0 0 128 0\nbuffer-32 0 0 32 0\nxHCI 1KB stream ctx arrays 0 0 1024 0\nxHCI 256 byte stream ctx arrays 0 0 256 0\nxHCI input/output contexts 15 20 2112 20\nxHCI ring segments 46 76 4096 76\nbuffer-2048 0 32 2048 16\nbuffer-512 0 32 512 4\nbuffer-128 3 32 128 1\nbuffer-32 0 128 32 1\n
|
|
||||||
A: power/control=on\n
|
|
||||||
A: power/runtime_active_time=837797789\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=0\n
|
|
||||||
A: power/wakeup=enabled\n
|
|
||||||
A: power/wakeup_abort_count=0\n
|
|
||||||
A: power/wakeup_active=0\n
|
|
||||||
A: power/wakeup_active_count=67\n
|
|
||||||
A: power/wakeup_count=0\n
|
|
||||||
A: power/wakeup_expire_count=67\n
|
|
||||||
A: power/wakeup_last_time_ms=835747082\n
|
|
||||||
A: power/wakeup_max_time_ms=108\n
|
|
||||||
A: power/wakeup_total_time_ms=6974\n
|
|
||||||
A: power_state=D0\n
|
|
||||||
A: resource=0x00000000b1320000 0x00000000b132ffff 0x0000000000140204\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n
|
|
||||||
A: revision=0x10\n
|
|
||||||
A: subsystem_device=0x312a\n
|
|
||||||
A: subsystem_vendor=0x17aa\n
|
|
||||||
A: vendor=0x8086\n
|
|
||||||
|
|
Binary file not shown.
Binary file not shown.
Before Width: | Height: | Size: 106 KiB |
|
@ -1,220 +0,0 @@
|
||||||
P: /devices/pci0000:00/0000:00:14.0/usb1/1-10
|
|
||||||
N: bus/usb/001/047=12010002000000405E04CA000001010203010902200001010080820904000002FFFFFF000705810340000807058202400000
|
|
||||||
E: DEVNAME=/dev/bus/usb/001/047
|
|
||||||
E: DEVTYPE=usb_device
|
|
||||||
E: DRIVER=usb
|
|
||||||
E: PRODUCT=45e/ca/100
|
|
||||||
E: TYPE=0/0/0
|
|
||||||
E: BUSNUM=001
|
|
||||||
E: DEVNUM=047
|
|
||||||
E: MAJOR=189
|
|
||||||
E: MINOR=46
|
|
||||||
E: SUBSYSTEM=usb
|
|
||||||
E: ID_VENDOR=Microsoft
|
|
||||||
E: ID_VENDOR_ENC=Microsoft
|
|
||||||
E: ID_VENDOR_ID=045e
|
|
||||||
E: ID_MODEL=Microsoft®_Fingerprint_Reader
|
|
||||||
E: ID_MODEL_ENC=Microsoft®\x20Fingerprint\x20Reader
|
|
||||||
E: ID_MODEL_ID=00ca
|
|
||||||
E: ID_REVISION=0100
|
|
||||||
E: ID_SERIAL=Microsoft_Microsoft®_Fingerprint_Reader__BE815DAD-15E4-0745-AA30-41DEBCAC5913_
|
|
||||||
E: ID_SERIAL_SHORT=_BE815DAD-15E4-0745-AA30-41DEBCAC5913_
|
|
||||||
E: ID_BUS=usb
|
|
||||||
E: ID_USB_INTERFACES=:ffffff:
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Microsoft Corp.
|
|
||||||
E: ID_AUTOSUSPEND=1
|
|
||||||
E: ID_MODEL_FROM_DATABASE=Fingerprint Reader
|
|
||||||
E: ID_PATH=pci-0000:00:14.0-usb-0:10
|
|
||||||
E: ID_PATH_TAG=pci-0000_00_14_0-usb-0_10
|
|
||||||
E: LIBFPRINT_DRIVER=Digital Persona U.are.U 4000/4000B/4500
|
|
||||||
E: ID_FOR_SEAT=usb-pci-0000_00_14_0-usb-0_10
|
|
||||||
E: TAGS=:seat:
|
|
||||||
E: CURRENT_TAGS=:seat:
|
|
||||||
A: authorized=1\n
|
|
||||||
A: avoid_reset_quirk=0\n
|
|
||||||
A: bConfigurationValue=1\n
|
|
||||||
A: bDeviceClass=00\n
|
|
||||||
A: bDeviceProtocol=00\n
|
|
||||||
A: bDeviceSubClass=00\n
|
|
||||||
A: bMaxPacketSize0=64\n
|
|
||||||
A: bMaxPower=260mA\n
|
|
||||||
A: bNumConfigurations=1\n
|
|
||||||
A: bNumInterfaces= 1\n
|
|
||||||
A: bcdDevice=0100\n
|
|
||||||
A: bmAttributes=80\n
|
|
||||||
A: busnum=1\n
|
|
||||||
A: configuration=
|
|
||||||
H: descriptors=12010002000000405E04CA000001010203010902200001010080820904000002FFFFFF000705810340000807058202400000
|
|
||||||
A: dev=189:46\n
|
|
||||||
A: devnum=47\n
|
|
||||||
A: devpath=10\n
|
|
||||||
L: driver=../../../../../bus/usb/drivers/usb
|
|
||||||
L: firmware_node=../../../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:4c/device:4d/device:57
|
|
||||||
A: idProduct=00ca\n
|
|
||||||
A: idVendor=045e\n
|
|
||||||
A: ltm_capable=no\n
|
|
||||||
A: manufacturer=Microsoft\n
|
|
||||||
A: maxchild=0\n
|
|
||||||
L: port=../1-0:1.0/usb1-port10
|
|
||||||
A: power/active_duration=31642\n
|
|
||||||
A: power/autosuspend=2\n
|
|
||||||
A: power/autosuspend_delay_ms=2000\n
|
|
||||||
A: power/connected_duration=1177852\n
|
|
||||||
A: power/control=auto\n
|
|
||||||
A: power/level=auto\n
|
|
||||||
A: power/persist=0\n
|
|
||||||
A: power/runtime_active_time=31877\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=1145731\n
|
|
||||||
A: product=Microsoft\302\256 Fingerprint Reader\n
|
|
||||||
A: quirks=0x0\n
|
|
||||||
A: removable=removable\n
|
|
||||||
A: rx_lanes=1\n
|
|
||||||
A: serial={BE815DAD-15E4-0745-AA30-41DEBCAC5913}\n
|
|
||||||
A: speed=12\n
|
|
||||||
A: tx_lanes=1\n
|
|
||||||
A: urbnum=183\n
|
|
||||||
A: version= 2.00\n
|
|
||||||
|
|
||||||
P: /devices/pci0000:00/0000:00:14.0/usb1
|
|
||||||
N: bus/usb/001/001=12010002090001406B1D020013050302010109021900010100E0000904000001090000000705810304000C
|
|
||||||
E: DEVNAME=/dev/bus/usb/001/001
|
|
||||||
E: DEVTYPE=usb_device
|
|
||||||
E: DRIVER=usb
|
|
||||||
E: PRODUCT=1d6b/2/513
|
|
||||||
E: TYPE=9/0/1
|
|
||||||
E: BUSNUM=001
|
|
||||||
E: DEVNUM=001
|
|
||||||
E: MAJOR=189
|
|
||||||
E: MINOR=0
|
|
||||||
E: SUBSYSTEM=usb
|
|
||||||
E: ID_VENDOR=Linux_5.13.12-200.fc34.x86_64_xhci-hcd
|
|
||||||
E: ID_VENDOR_ENC=Linux\x205.13.12-200.fc34.x86_64\x20xhci-hcd
|
|
||||||
E: ID_VENDOR_ID=1d6b
|
|
||||||
E: ID_MODEL=xHCI_Host_Controller
|
|
||||||
E: ID_MODEL_ENC=xHCI\x20Host\x20Controller
|
|
||||||
E: ID_MODEL_ID=0002
|
|
||||||
E: ID_REVISION=0513
|
|
||||||
E: ID_SERIAL=Linux_5.13.12-200.fc34.x86_64_xhci-hcd_xHCI_Host_Controller_0000:00:14.0
|
|
||||||
E: ID_SERIAL_SHORT=0000:00:14.0
|
|
||||||
E: ID_BUS=usb
|
|
||||||
E: ID_USB_INTERFACES=:090000:
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Linux Foundation
|
|
||||||
E: ID_AUTOSUSPEND=1
|
|
||||||
E: ID_MODEL_FROM_DATABASE=2.0 root hub
|
|
||||||
E: ID_PATH=pci-0000:00:14.0
|
|
||||||
E: ID_PATH_TAG=pci-0000_00_14_0
|
|
||||||
E: ID_FOR_SEAT=usb-pci-0000_00_14_0
|
|
||||||
E: TAGS=:seat:
|
|
||||||
E: CURRENT_TAGS=:seat:
|
|
||||||
A: authorized=1\n
|
|
||||||
A: authorized_default=1\n
|
|
||||||
A: avoid_reset_quirk=0\n
|
|
||||||
A: bConfigurationValue=1\n
|
|
||||||
A: bDeviceClass=09\n
|
|
||||||
A: bDeviceProtocol=01\n
|
|
||||||
A: bDeviceSubClass=00\n
|
|
||||||
A: bMaxPacketSize0=64\n
|
|
||||||
A: bMaxPower=0mA\n
|
|
||||||
A: bNumConfigurations=1\n
|
|
||||||
A: bNumInterfaces= 1\n
|
|
||||||
A: bcdDevice=0513\n
|
|
||||||
A: bmAttributes=e0\n
|
|
||||||
A: busnum=1\n
|
|
||||||
A: configuration=
|
|
||||||
H: descriptors=12010002090001406B1D020013050302010109021900010100E0000904000001090000000705810304000C
|
|
||||||
A: dev=189:0\n
|
|
||||||
A: devnum=1\n
|
|
||||||
A: devpath=0\n
|
|
||||||
L: driver=../../../../bus/usb/drivers/usb
|
|
||||||
L: firmware_node=../../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:4c/device:4d
|
|
||||||
A: idProduct=0002\n
|
|
||||||
A: idVendor=1d6b\n
|
|
||||||
A: interface_authorized_default=1\n
|
|
||||||
A: ltm_capable=no\n
|
|
||||||
A: manufacturer=Linux 5.13.12-200.fc34.x86_64 xhci-hcd\n
|
|
||||||
A: maxchild=16\n
|
|
||||||
A: power/active_duration=775798957\n
|
|
||||||
A: power/autosuspend=0\n
|
|
||||||
A: power/autosuspend_delay_ms=0\n
|
|
||||||
A: power/connected_duration=775798957\n
|
|
||||||
A: power/control=auto\n
|
|
||||||
A: power/level=auto\n
|
|
||||||
A: power/runtime_active_time=775798954\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=0\n
|
|
||||||
A: power/wakeup=disabled\n
|
|
||||||
A: power/wakeup_abort_count=\n
|
|
||||||
A: power/wakeup_active=\n
|
|
||||||
A: power/wakeup_active_count=\n
|
|
||||||
A: power/wakeup_count=\n
|
|
||||||
A: power/wakeup_expire_count=\n
|
|
||||||
A: power/wakeup_last_time_ms=\n
|
|
||||||
A: power/wakeup_max_time_ms=\n
|
|
||||||
A: power/wakeup_total_time_ms=\n
|
|
||||||
A: product=xHCI Host Controller\n
|
|
||||||
A: quirks=0x0\n
|
|
||||||
A: removable=unknown\n
|
|
||||||
A: rx_lanes=1\n
|
|
||||||
A: serial=0000:00:14.0\n
|
|
||||||
A: speed=480\n
|
|
||||||
A: tx_lanes=1\n
|
|
||||||
A: urbnum=1381\n
|
|
||||||
A: version= 2.00\n
|
|
||||||
|
|
||||||
P: /devices/pci0000:00/0000:00:14.0
|
|
||||||
E: DRIVER=xhci_hcd
|
|
||||||
E: PCI_CLASS=C0330
|
|
||||||
E: PCI_ID=8086:A36D
|
|
||||||
E: PCI_SUBSYS_ID=17AA:312A
|
|
||||||
E: PCI_SLOT_NAME=0000:00:14.0
|
|
||||||
E: MODALIAS=pci:v00008086d0000A36Dsv000017AAsd0000312Abc0Csc03i30
|
|
||||||
E: SUBSYSTEM=pci
|
|
||||||
E: ID_PCI_CLASS_FROM_DATABASE=Serial bus controller
|
|
||||||
E: ID_PCI_SUBCLASS_FROM_DATABASE=USB controller
|
|
||||||
E: ID_PCI_INTERFACE_FROM_DATABASE=XHCI
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Intel Corporation
|
|
||||||
E: ID_MODEL_FROM_DATABASE=Cannon Lake PCH USB 3.1 xHCI Host Controller
|
|
||||||
A: ari_enabled=0\n
|
|
||||||
A: broken_parity_status=0\n
|
|
||||||
A: class=0x0c0330\n
|
|
||||||
H: config=86806DA3060490021030030C00008000040032B1000000000000000000000000000000000000000000000000AA172A31000000007000000000000000FF010000FD0134808FC6FF8300000000000000007F6DDC0F00000000970B083900000000316000000000000000000000000000000180C2C1080000000000000000000000059087009802E0FE0000000000000000090014F01000400100000000C10A080000080E00001800008F400200000100006000000008020000010000000400000090000000001800000005000000000000000300000C0000004000000080000000030000000000000000000000000000000000000000000000B50F110112000000
|
|
||||||
A: consistent_dma_mask_bits=64\n
|
|
||||||
A: d3cold_allowed=1\n
|
|
||||||
A: dbc=disabled\n
|
|
||||||
A: device=0xa36d\n
|
|
||||||
A: dma_mask_bits=64\n
|
|
||||||
L: driver=../../../bus/pci/drivers/xhci_hcd
|
|
||||||
A: driver_override=(null)\n
|
|
||||||
A: enable=1\n
|
|
||||||
L: firmware_node=../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:4c
|
|
||||||
A: index=3\n
|
|
||||||
A: irq=125\n
|
|
||||||
A: label=Onboard - Other\n
|
|
||||||
A: local_cpulist=0-5\n
|
|
||||||
A: local_cpus=3f\n
|
|
||||||
A: modalias=pci:v00008086d0000A36Dsv000017AAsd0000312Abc0Csc03i30\n
|
|
||||||
A: msi_bus=1\n
|
|
||||||
A: msi_irqs/125=msi\n
|
|
||||||
A: numa_node=-1\n
|
|
||||||
A: pools=poolinfo - 0.1\nbuffer-2048 0 0 2048 0\nbuffer-512 0 0 512 0\nbuffer-128 0 0 128 0\nbuffer-32 0 0 32 0\nxHCI 1KB stream ctx arrays 0 0 1024 0\nxHCI 256 byte stream ctx arrays 0 0 256 0\nxHCI input/output contexts 15 20 2112 20\nxHCI ring segments 46 76 4096 76\nbuffer-2048 0 32 2048 16\nbuffer-512 0 32 512 4\nbuffer-128 3 32 128 1\nbuffer-32 0 128 32 1\n
|
|
||||||
A: power/control=on\n
|
|
||||||
A: power/runtime_active_time=775799103\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=0\n
|
|
||||||
A: power/wakeup=enabled\n
|
|
||||||
A: power/wakeup_abort_count=0\n
|
|
||||||
A: power/wakeup_active=0\n
|
|
||||||
A: power/wakeup_active_count=61\n
|
|
||||||
A: power/wakeup_count=0\n
|
|
||||||
A: power/wakeup_expire_count=61\n
|
|
||||||
A: power/wakeup_last_time_ms=773160682\n
|
|
||||||
A: power/wakeup_max_time_ms=108\n
|
|
||||||
A: power/wakeup_total_time_ms=6358\n
|
|
||||||
A: power_state=D0\n
|
|
||||||
A: resource=0x00000000b1320000 0x00000000b132ffff 0x0000000000140204\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n
|
|
||||||
A: revision=0x10\n
|
|
||||||
A: subsystem_device=0x312a\n
|
|
||||||
A: subsystem_vendor=0x17aa\n
|
|
||||||
A: vendor=0x8086\n
|
|
||||||
|
|
|
@ -81,145 +81,3 @@ A: speed=12
|
||||||
A: tx_lanes=1
|
A: tx_lanes=1
|
||||||
A: urbnum=8
|
A: urbnum=8
|
||||||
A: version= 1.10
|
A: version= 1.10
|
||||||
|
|
||||||
P: /devices/pci0000:00/0000:00:14.0/usb1
|
|
||||||
N: bus/usb/001/001=12010002090001406B1D020011050302010109021900010100E0000904000001090000000705810304000C
|
|
||||||
E: DEVNAME=/dev/bus/usb/001/001
|
|
||||||
E: DEVTYPE=usb_device
|
|
||||||
E: DRIVER=usb
|
|
||||||
E: PRODUCT=1d6b/2/511
|
|
||||||
E: TYPE=9/0/1
|
|
||||||
E: BUSNUM=001
|
|
||||||
E: DEVNUM=001
|
|
||||||
E: MAJOR=189
|
|
||||||
E: MINOR=0
|
|
||||||
E: SUBSYSTEM=usb
|
|
||||||
E: ID_VENDOR=Linux_5.11.2-arch1-1_xhci-hcd
|
|
||||||
E: ID_VENDOR_ENC=Linux\x205.11.2-arch1-1\x20xhci-hcd
|
|
||||||
E: ID_VENDOR_ID=1d6b
|
|
||||||
E: ID_MODEL=xHCI_Host_Controller
|
|
||||||
E: ID_MODEL_ENC=xHCI\x20Host\x20Controller
|
|
||||||
E: ID_MODEL_ID=0002
|
|
||||||
E: ID_REVISION=0511
|
|
||||||
E: ID_SERIAL=Linux_5.11.2-arch1-1_xhci-hcd_xHCI_Host_Controller_0000:00:14.0
|
|
||||||
E: ID_SERIAL_SHORT=0000:00:14.0
|
|
||||||
E: ID_BUS=usb
|
|
||||||
E: ID_USB_INTERFACES=:090000:
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Linux Foundation
|
|
||||||
E: ID_AUTOSUSPEND=1
|
|
||||||
E: ID_MODEL_FROM_DATABASE=2.0 root hub
|
|
||||||
E: ID_PATH=pci-0000:00:14.0
|
|
||||||
E: ID_PATH_TAG=pci-0000_00_14_0
|
|
||||||
E: ID_FOR_SEAT=usb-pci-0000_00_14_0
|
|
||||||
E: TAGS=:seat:
|
|
||||||
E: CURRENT_TAGS=:seat:
|
|
||||||
A: authorized=1\n
|
|
||||||
A: authorized_default=1\n
|
|
||||||
A: avoid_reset_quirk=0\n
|
|
||||||
A: bConfigurationValue=1\n
|
|
||||||
A: bDeviceClass=09\n
|
|
||||||
A: bDeviceProtocol=01\n
|
|
||||||
A: bDeviceSubClass=00\n
|
|
||||||
A: bMaxPacketSize0=64\n
|
|
||||||
A: bMaxPower=0mA\n
|
|
||||||
A: bNumConfigurations=1\n
|
|
||||||
A: bNumInterfaces= 1\n
|
|
||||||
A: bcdDevice=0511\n
|
|
||||||
A: bmAttributes=e0\n
|
|
||||||
A: busnum=1\n
|
|
||||||
A: configuration=
|
|
||||||
H: descriptors=12010002090001406B1D020011050302010109021900010100E0000904000001090000000705810304000C
|
|
||||||
A: dev=189:0\n
|
|
||||||
A: devnum=1\n
|
|
||||||
A: devpath=0\n
|
|
||||||
L: driver=../../../../bus/usb/drivers/usb
|
|
||||||
L: firmware_node=../../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:4c/device:4d
|
|
||||||
A: idProduct=0002\n
|
|
||||||
A: idVendor=1d6b\n
|
|
||||||
A: interface_authorized_default=1\n
|
|
||||||
A: ltm_capable=no\n
|
|
||||||
A: manufacturer=Linux 5.11.2-arch1-1 xhci-hcd\n
|
|
||||||
A: maxchild=16\n
|
|
||||||
A: power/active_duration=3289930\n
|
|
||||||
A: power/autosuspend=0\n
|
|
||||||
A: power/autosuspend_delay_ms=0\n
|
|
||||||
A: power/connected_duration=34389654\n
|
|
||||||
A: power/control=auto\n
|
|
||||||
A: power/level=auto\n
|
|
||||||
A: power/runtime_active_time=3289845\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=31099805\n
|
|
||||||
A: power/wakeup=disabled\n
|
|
||||||
A: power/wakeup_abort_count=\n
|
|
||||||
A: power/wakeup_active=\n
|
|
||||||
A: power/wakeup_active_count=\n
|
|
||||||
A: power/wakeup_count=\n
|
|
||||||
A: power/wakeup_expire_count=\n
|
|
||||||
A: power/wakeup_last_time_ms=\n
|
|
||||||
A: power/wakeup_max_time_ms=\n
|
|
||||||
A: power/wakeup_total_time_ms=\n
|
|
||||||
A: product=xHCI Host Controller\n
|
|
||||||
A: quirks=0x0\n
|
|
||||||
A: removable=unknown\n
|
|
||||||
A: rx_lanes=1\n
|
|
||||||
A: serial=0000:00:14.0\n
|
|
||||||
A: speed=480\n
|
|
||||||
A: tx_lanes=1\n
|
|
||||||
A: urbnum=2355\n
|
|
||||||
A: version= 2.00\n
|
|
||||||
|
|
||||||
P: /devices/pci0000:00/0000:00:14.0
|
|
||||||
E: DRIVER=xhci_hcd
|
|
||||||
E: PCI_CLASS=C0330
|
|
||||||
E: PCI_ID=8086:A12F
|
|
||||||
E: PCI_SUBSYS_ID=1028:07BE
|
|
||||||
E: PCI_SLOT_NAME=0000:00:14.0
|
|
||||||
E: MODALIAS=pci:v00008086d0000A12Fsv00001028sd000007BEbc0Csc03i30
|
|
||||||
E: SUBSYSTEM=pci
|
|
||||||
E: ID_PCI_CLASS_FROM_DATABASE=Serial bus controller
|
|
||||||
E: ID_PCI_SUBCLASS_FROM_DATABASE=USB controller
|
|
||||||
E: ID_PCI_INTERFACE_FROM_DATABASE=XHCI
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Intel Corporation
|
|
||||||
E: ID_MODEL_FROM_DATABASE=100 Series/C230 Series Chipset Family USB 3.0 xHCI Controller
|
|
||||||
A: ari_enabled=0\n
|
|
||||||
A: broken_parity_status=0\n
|
|
||||||
A: class=0x0c0330\n
|
|
||||||
H: config=86802FA1060490023130030C000080000400D1ED0000000000000000000000000000000000000000000000002810BE07000000007000000000000000FF010000
|
|
||||||
A: consistent_dma_mask_bits=64\n
|
|
||||||
A: d3cold_allowed=1\n
|
|
||||||
A: device=0xa12f\n
|
|
||||||
A: dma_mask_bits=64\n
|
|
||||||
L: driver=../../../bus/pci/drivers/xhci_hcd
|
|
||||||
A: driver_override=(null)\n
|
|
||||||
A: enable=1\n
|
|
||||||
L: firmware_node=../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:4c
|
|
||||||
L: iommu=../../virtual/iommu/dmar1
|
|
||||||
L: iommu_group=../../../kernel/iommu_groups/4
|
|
||||||
A: irq=143\n
|
|
||||||
A: local_cpulist=0-7\n
|
|
||||||
A: local_cpus=ff\n
|
|
||||||
A: modalias=pci:v00008086d0000A12Fsv00001028sd000007BEbc0Csc03i30\n
|
|
||||||
A: msi_bus=1\n
|
|
||||||
A: msi_irqs/143=msi\n
|
|
||||||
A: numa_node=-1\n
|
|
||||||
A: pools=poolinfo - 0.1\nbuffer-2048 0 0 2048 0\nbuffer-512 0 0 512 0\nbuffer-128 0 0 128 0\nbuffer-32 0 0 32 0\nxHCI 1KB stream ctx arrays 0 0 1024 0\nxHCI 256 byte stream ctx arrays 0 0 256 0\nxHCI input/output contexts 7 10 2112 10\nxHCI ring segments 30 38 4096 38\nbuffer-2048 0 0 2048 0\nbuffer-512 0 0 512 0\nbuffer-128 0 0 128 0\nbuffer-32 0 0 32 0\n
|
|
||||||
A: power/control=on\n
|
|
||||||
A: power/runtime_active_time=34390988\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=0\n
|
|
||||||
A: power/wakeup=enabled\n
|
|
||||||
A: power/wakeup_abort_count=0\n
|
|
||||||
A: power/wakeup_active=0\n
|
|
||||||
A: power/wakeup_active_count=0\n
|
|
||||||
A: power/wakeup_count=0\n
|
|
||||||
A: power/wakeup_expire_count=0\n
|
|
||||||
A: power/wakeup_last_time_ms=0\n
|
|
||||||
A: power/wakeup_max_time_ms=0\n
|
|
||||||
A: power/wakeup_total_time_ms=0\n
|
|
||||||
A: power_state=D0\n
|
|
||||||
A: resource=0x00000000edd10000 0x00000000edd1ffff 0x0000000000140204\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n
|
|
||||||
A: revision=0x31\n
|
|
||||||
A: subsystem_device=0x07be\n
|
|
||||||
A: subsystem_vendor=0x1028\n
|
|
||||||
A: vendor=0x8086\n
|
|
||||||
|
|
||||||
|
|
|
@ -78,145 +78,3 @@ A: speed=12
|
||||||
A: tx_lanes=1
|
A: tx_lanes=1
|
||||||
A: urbnum=7
|
A: urbnum=7
|
||||||
A: version= 1.10
|
A: version= 1.10
|
||||||
|
|
||||||
P: /devices/pci0000:00/0000:00:14.0/usb2
|
|
||||||
N: bus/usb/002/001=12010002090001406B1D020011050302010109021900010100E0000904000001090000000705810304000C
|
|
||||||
E: DEVNAME=/dev/bus/usb/002/001
|
|
||||||
E: DEVTYPE=usb_device
|
|
||||||
E: DRIVER=usb
|
|
||||||
E: PRODUCT=1d6b/2/511
|
|
||||||
E: TYPE=9/0/1
|
|
||||||
E: BUSNUM=002
|
|
||||||
E: DEVNUM=001
|
|
||||||
E: MAJOR=189
|
|
||||||
E: MINOR=0
|
|
||||||
E: SUBSYSTEM=usb
|
|
||||||
E: ID_VENDOR=Linux_5.11.2-arch1-1_xhci-hcd
|
|
||||||
E: ID_VENDOR_ENC=Linux\x205.11.2-arch1-1\x20xhci-hcd
|
|
||||||
E: ID_VENDOR_ID=1d6b
|
|
||||||
E: ID_MODEL=xHCI_Host_Controller
|
|
||||||
E: ID_MODEL_ENC=xHCI\x20Host\x20Controller
|
|
||||||
E: ID_MODEL_ID=0002
|
|
||||||
E: ID_REVISION=0511
|
|
||||||
E: ID_SERIAL=Linux_5.11.2-arch1-1_xhci-hcd_xHCI_Host_Controller_0000:00:14.0
|
|
||||||
E: ID_SERIAL_SHORT=0000:00:14.0
|
|
||||||
E: ID_BUS=usb
|
|
||||||
E: ID_USB_INTERFACES=:090000:
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Linux Foundation
|
|
||||||
E: ID_AUTOSUSPEND=1
|
|
||||||
E: ID_MODEL_FROM_DATABASE=2.0 root hub
|
|
||||||
E: ID_PATH=pci-0000:00:14.0
|
|
||||||
E: ID_PATH_TAG=pci-0000_00_14_0
|
|
||||||
E: ID_FOR_SEAT=usb-pci-0000_00_14_0
|
|
||||||
E: TAGS=:seat:
|
|
||||||
E: CURRENT_TAGS=:seat:
|
|
||||||
A: authorized=1\n
|
|
||||||
A: authorized_default=1\n
|
|
||||||
A: avoid_reset_quirk=0\n
|
|
||||||
A: bConfigurationValue=1\n
|
|
||||||
A: bDeviceClass=09\n
|
|
||||||
A: bDeviceProtocol=01\n
|
|
||||||
A: bDeviceSubClass=00\n
|
|
||||||
A: bMaxPacketSize0=64\n
|
|
||||||
A: bMaxPower=0mA\n
|
|
||||||
A: bNumConfigurations=1\n
|
|
||||||
A: bNumInterfaces= 1\n
|
|
||||||
A: bcdDevice=0511\n
|
|
||||||
A: bmAttributes=e0\n
|
|
||||||
A: busnum=1\n
|
|
||||||
A: configuration=
|
|
||||||
H: descriptors=12010002090001406B1D020011050302010109021900010100E0000904000001090000000705810304000C
|
|
||||||
A: dev=189:0\n
|
|
||||||
A: devnum=1\n
|
|
||||||
A: devpath=0\n
|
|
||||||
L: driver=../../../../bus/usb/drivers/usb
|
|
||||||
L: firmware_node=../../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:4c/device:4d
|
|
||||||
A: idProduct=0002\n
|
|
||||||
A: idVendor=1d6b\n
|
|
||||||
A: interface_authorized_default=1\n
|
|
||||||
A: ltm_capable=no\n
|
|
||||||
A: manufacturer=Linux 5.11.2-arch1-1 xhci-hcd\n
|
|
||||||
A: maxchild=16\n
|
|
||||||
A: power/active_duration=3289930\n
|
|
||||||
A: power/autosuspend=0\n
|
|
||||||
A: power/autosuspend_delay_ms=0\n
|
|
||||||
A: power/connected_duration=34389654\n
|
|
||||||
A: power/control=auto\n
|
|
||||||
A: power/level=auto\n
|
|
||||||
A: power/runtime_active_time=3289845\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=31099805\n
|
|
||||||
A: power/wakeup=disabled\n
|
|
||||||
A: power/wakeup_abort_count=\n
|
|
||||||
A: power/wakeup_active=\n
|
|
||||||
A: power/wakeup_active_count=\n
|
|
||||||
A: power/wakeup_count=\n
|
|
||||||
A: power/wakeup_expire_count=\n
|
|
||||||
A: power/wakeup_last_time_ms=\n
|
|
||||||
A: power/wakeup_max_time_ms=\n
|
|
||||||
A: power/wakeup_total_time_ms=\n
|
|
||||||
A: product=xHCI Host Controller\n
|
|
||||||
A: quirks=0x0\n
|
|
||||||
A: removable=unknown\n
|
|
||||||
A: rx_lanes=1\n
|
|
||||||
A: serial=0000:00:14.0\n
|
|
||||||
A: speed=480\n
|
|
||||||
A: tx_lanes=1\n
|
|
||||||
A: urbnum=2355\n
|
|
||||||
A: version= 2.00\n
|
|
||||||
|
|
||||||
P: /devices/pci0000:00/0000:00:14.0
|
|
||||||
E: DRIVER=xhci_hcd
|
|
||||||
E: PCI_CLASS=C0330
|
|
||||||
E: PCI_ID=8086:A12F
|
|
||||||
E: PCI_SUBSYS_ID=1028:07BE
|
|
||||||
E: PCI_SLOT_NAME=0000:00:14.0
|
|
||||||
E: MODALIAS=pci:v00008086d0000A12Fsv00001028sd000007BEbc0Csc03i30
|
|
||||||
E: SUBSYSTEM=pci
|
|
||||||
E: ID_PCI_CLASS_FROM_DATABASE=Serial bus controller
|
|
||||||
E: ID_PCI_SUBCLASS_FROM_DATABASE=USB controller
|
|
||||||
E: ID_PCI_INTERFACE_FROM_DATABASE=XHCI
|
|
||||||
E: ID_VENDOR_FROM_DATABASE=Intel Corporation
|
|
||||||
E: ID_MODEL_FROM_DATABASE=100 Series/C230 Series Chipset Family USB 3.0 xHCI Controller
|
|
||||||
A: ari_enabled=0\n
|
|
||||||
A: broken_parity_status=0\n
|
|
||||||
A: class=0x0c0330\n
|
|
||||||
H: config=86802FA1060490023130030C000080000400D1ED0000000000000000000000000000000000000000000000002810BE07000000007000000000000000FF010000
|
|
||||||
A: consistent_dma_mask_bits=64\n
|
|
||||||
A: d3cold_allowed=1\n
|
|
||||||
A: device=0xa12f\n
|
|
||||||
A: dma_mask_bits=64\n
|
|
||||||
L: driver=../../../bus/pci/drivers/xhci_hcd
|
|
||||||
A: driver_override=(null)\n
|
|
||||||
A: enable=1\n
|
|
||||||
L: firmware_node=../../LNXSYSTM:00/LNXSYBUS:00/PNP0A08:00/device:4c
|
|
||||||
L: iommu=../../virtual/iommu/dmar1
|
|
||||||
L: iommu_group=../../../kernel/iommu_groups/4
|
|
||||||
A: irq=143\n
|
|
||||||
A: local_cpulist=0-7\n
|
|
||||||
A: local_cpus=ff\n
|
|
||||||
A: modalias=pci:v00008086d0000A12Fsv00001028sd000007BEbc0Csc03i30\n
|
|
||||||
A: msi_bus=1\n
|
|
||||||
A: msi_irqs/143=msi\n
|
|
||||||
A: numa_node=-1\n
|
|
||||||
A: pools=poolinfo - 0.1\nbuffer-2048 0 0 2048 0\nbuffer-512 0 0 512 0\nbuffer-128 0 0 128 0\nbuffer-32 0 0 32 0\nxHCI 1KB stream ctx arrays 0 0 1024 0\nxHCI 256 byte stream ctx arrays 0 0 256 0\nxHCI input/output contexts 7 10 2112 10\nxHCI ring segments 30 38 4096 38\nbuffer-2048 0 0 2048 0\nbuffer-512 0 0 512 0\nbuffer-128 0 0 128 0\nbuffer-32 0 0 32 0\n
|
|
||||||
A: power/control=on\n
|
|
||||||
A: power/runtime_active_time=34390988\n
|
|
||||||
A: power/runtime_status=active\n
|
|
||||||
A: power/runtime_suspended_time=0\n
|
|
||||||
A: power/wakeup=enabled\n
|
|
||||||
A: power/wakeup_abort_count=0\n
|
|
||||||
A: power/wakeup_active=0\n
|
|
||||||
A: power/wakeup_active_count=0\n
|
|
||||||
A: power/wakeup_count=0\n
|
|
||||||
A: power/wakeup_expire_count=0\n
|
|
||||||
A: power/wakeup_last_time_ms=0\n
|
|
||||||
A: power/wakeup_max_time_ms=0\n
|
|
||||||
A: power/wakeup_total_time_ms=0\n
|
|
||||||
A: power_state=D0\n
|
|
||||||
A: resource=0x00000000edd10000 0x00000000edd1ffff 0x0000000000140204\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n0x0000000000000000 0x0000000000000000 0x0000000000000000\n
|
|
||||||
A: revision=0x31\n
|
|
||||||
A: subsystem_device=0x07be\n
|
|
||||||
A: subsystem_vendor=0x1028\n
|
|
||||||
A: vendor=0x8086\n
|
|
||||||
|
|
||||||
|
|
|
@ -322,7 +322,7 @@ class VirtualDeviceBase(unittest.TestCase):
|
||||||
else:
|
else:
|
||||||
self.assertFalse(match)
|
self.assertFalse(match)
|
||||||
|
|
||||||
if isinstance(scan_nick, str) and not self.dev.has_storage():
|
if isinstance(scan_nick, str):
|
||||||
self.assertEqual(self._verify_fp.props.fpi_data.get_string(), scan_nick)
|
self.assertEqual(self._verify_fp.props.fpi_data.get_string(), scan_nick)
|
||||||
|
|
||||||
|
|
||||||
|
@ -470,6 +470,13 @@ class VirtualDevice(VirtualDeviceBase):
|
||||||
def test_enroll_verify_no_match(self):
|
def test_enroll_verify_no_match(self):
|
||||||
matching = self.enroll_print('testprint', FPrint.Finger.LEFT_RING)
|
matching = self.enroll_print('testprint', FPrint.Finger.LEFT_RING)
|
||||||
|
|
||||||
|
if self.dev.has_storage():
|
||||||
|
with self.assertRaises(GLib.Error) as error:
|
||||||
|
self.check_verify(matching, 'not-testprint', match=False,
|
||||||
|
identify=self.dev.supports_identify())
|
||||||
|
self.assertTrue(error.exception.matches(FPrint.DeviceError.quark(),
|
||||||
|
FPrint.DeviceError.DATA_NOT_FOUND))
|
||||||
|
else:
|
||||||
self.check_verify(matching, 'not-testprint', match=False,
|
self.check_verify(matching, 'not-testprint', match=False,
|
||||||
identify=self.dev.supports_identify())
|
identify=self.dev.supports_identify())
|
||||||
|
|
||||||
|
@ -590,11 +597,14 @@ class VirtualDevice(VirtualDeviceBase):
|
||||||
FPrint.DeviceRetry.TOO_SHORT))
|
FPrint.DeviceRetry.TOO_SHORT))
|
||||||
|
|
||||||
self.send_command('SCAN', 'another-id')
|
self.send_command('SCAN', 'another-id')
|
||||||
|
if self.dev.has_storage():
|
||||||
|
with self.assertRaises(GLib.GError) as error:
|
||||||
|
self.dev.verify_sync(enrolled)
|
||||||
|
self.assertTrue(error.exception.matches(FPrint.DeviceError.quark(),
|
||||||
|
FPrint.DeviceError.DATA_NOT_FOUND))
|
||||||
|
else:
|
||||||
verify_match, verify_fp = self.dev.verify_sync(enrolled)
|
verify_match, verify_fp = self.dev.verify_sync(enrolled)
|
||||||
self.assertFalse(verify_match)
|
self.assertFalse(verify_match)
|
||||||
if self.dev.has_storage():
|
|
||||||
self.assertIsNone(verify_fp)
|
|
||||||
else:
|
|
||||||
self.assertFalse(verify_fp.equal(enrolled))
|
self.assertFalse(verify_fp.equal(enrolled))
|
||||||
|
|
||||||
self.send_auto(enrolled)
|
self.send_auto(enrolled)
|
||||||
|
@ -811,6 +821,12 @@ class VirtualDevice(VirtualDeviceBase):
|
||||||
self.wait_timeout(10)
|
self.wait_timeout(10)
|
||||||
self.assertFalse(self._verify_completed)
|
self.assertFalse(self._verify_completed)
|
||||||
|
|
||||||
|
if self.dev.has_storage():
|
||||||
|
with self.assertRaises(GLib.Error) as error:
|
||||||
|
self.complete_verify()
|
||||||
|
self.assertTrue(error.exception.matches(FPrint.DeviceError.quark(),
|
||||||
|
FPrint.DeviceError.DATA_NOT_FOUND))
|
||||||
|
else:
|
||||||
self.complete_verify()
|
self.complete_verify()
|
||||||
self.assertTrue(self._verify_reported)
|
self.assertTrue(self._verify_reported)
|
||||||
|
|
||||||
|
@ -1143,12 +1159,18 @@ class VirtualDeviceStorage(VirtualDevice):
|
||||||
FPrint.DeviceError.DATA_NOT_FOUND))
|
FPrint.DeviceError.DATA_NOT_FOUND))
|
||||||
|
|
||||||
def test_verify_missing_print(self):
|
def test_verify_missing_print(self):
|
||||||
|
with self.assertRaises(GLib.Error) as error:
|
||||||
self.check_verify(FPrint.Print.new(self.dev),
|
self.check_verify(FPrint.Print.new(self.dev),
|
||||||
'not-existing-print', False, identify=False)
|
'not-existing-print', False, identify=False)
|
||||||
|
self.assertTrue(error.exception.matches(FPrint.DeviceError.quark(),
|
||||||
|
FPrint.DeviceError.DATA_NOT_FOUND))
|
||||||
|
|
||||||
def test_identify_missing_print(self):
|
def test_identify_missing_print(self):
|
||||||
|
with self.assertRaises(GLib.Error) as error:
|
||||||
self.check_verify(FPrint.Print.new(self.dev),
|
self.check_verify(FPrint.Print.new(self.dev),
|
||||||
'not-existing-print', False, identify=True)
|
'not-existing-print', False, identify=True)
|
||||||
|
self.assertTrue(error.exception.matches(FPrint.DeviceError.quark(),
|
||||||
|
FPrint.DeviceError.DATA_NOT_FOUND))
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
|
|
144
tests/virtual-sdcp.py
Normal file
144
tests/virtual-sdcp.py
Normal file
|
@ -0,0 +1,144 @@
|
||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import sys
|
||||||
|
try:
|
||||||
|
import gi
|
||||||
|
import os
|
||||||
|
|
||||||
|
from gi.repository import GLib, Gio
|
||||||
|
|
||||||
|
import unittest
|
||||||
|
import subprocess
|
||||||
|
import shutil
|
||||||
|
import tempfile
|
||||||
|
except Exception as e:
|
||||||
|
print("Missing dependencies: %s" % str(e))
|
||||||
|
sys.exit(77)
|
||||||
|
|
||||||
|
FPrint = None
|
||||||
|
|
||||||
|
# Re-run the test with the passed wrapper if set
|
||||||
|
wrapper = os.getenv('LIBFPRINT_TEST_WRAPPER')
|
||||||
|
if wrapper:
|
||||||
|
wrap_cmd = wrapper.split(' ') + [sys.executable, os.path.abspath(__file__)] + \
|
||||||
|
sys.argv[1:]
|
||||||
|
os.unsetenv('LIBFPRINT_TEST_WRAPPER')
|
||||||
|
sys.exit(subprocess.check_call(wrap_cmd))
|
||||||
|
|
||||||
|
# Only permit loading virtual_sdcp driver for tests in this file
|
||||||
|
os.environ['FP_DRIVERS_WHITELIST'] = 'virtual_sdcp'
|
||||||
|
|
||||||
|
if hasattr(os.environ, 'MESON_SOURCE_ROOT'):
|
||||||
|
root = os.environ['MESON_SOURCE_ROOT']
|
||||||
|
else:
|
||||||
|
root = os.path.join(os.path.dirname(__file__), '..')
|
||||||
|
|
||||||
|
ctx = GLib.main_context_default()
|
||||||
|
|
||||||
|
class VirtualSDCP(unittest.TestCase):
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def setUpClass(cls):
|
||||||
|
os.environ['FP_VIRTUAL_SDCP'] = os.environ['SDCP_VIRT_BINARY']
|
||||||
|
|
||||||
|
cls.ctx = FPrint.Context()
|
||||||
|
|
||||||
|
cls.dev = None
|
||||||
|
for dev in cls.ctx.get_devices():
|
||||||
|
cls.dev = dev
|
||||||
|
break
|
||||||
|
|
||||||
|
assert cls.dev is not None, "You need to compile with virtual_sdcp for testing"
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def tearDownClass(cls):
|
||||||
|
del cls.dev
|
||||||
|
del cls.ctx
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def tearDown(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def enroll(self, progress_cb=None):
|
||||||
|
# Enroll another print
|
||||||
|
template = FPrint.Print.new(self.dev)
|
||||||
|
template.props.finger = FPrint.Finger.LEFT_THUMB
|
||||||
|
template.props.username = "testuser"
|
||||||
|
template.props.description = "test print"
|
||||||
|
datetime = GLib.DateTime.new_now_local()
|
||||||
|
date = GLib.Date()
|
||||||
|
date.set_dmy(*datetime.get_ymd()[::-1])
|
||||||
|
template.props.enroll_date = date
|
||||||
|
return self.dev.enroll_sync(template, None, progress_cb, None)
|
||||||
|
|
||||||
|
def test_connect(self):
|
||||||
|
self.dev.open_sync()
|
||||||
|
self.dev.close_sync()
|
||||||
|
|
||||||
|
def test_reconnect(self):
|
||||||
|
# Ensure device was opened once before, this may be a reconnect if
|
||||||
|
# it is the same process as another test.
|
||||||
|
self.dev.open_sync()
|
||||||
|
self.dev.close_sync()
|
||||||
|
|
||||||
|
# Check that a reconnect happens on next open. To know about this, we
|
||||||
|
# need to parse check log messages for that.
|
||||||
|
success = [False]
|
||||||
|
def log_func(domain, level, msg):
|
||||||
|
print("log: '%s', '%s', '%s'" % (str(domain), str(level), msg))
|
||||||
|
if msg == 'Reconnect succeeded':
|
||||||
|
success[0] = True
|
||||||
|
|
||||||
|
# Call default handler
|
||||||
|
GLib.log_default_handler(domain, level, msg)
|
||||||
|
|
||||||
|
handler_id = GLib.log_set_handler('libfprint-sdcp_device', GLib.LogLevelFlags.LEVEL_DEBUG, log_func)
|
||||||
|
self.dev.open_sync()
|
||||||
|
self.dev.close_sync()
|
||||||
|
GLib.log_remove_handler('libfprint-sdcp_device', handler_id)
|
||||||
|
assert success[0]
|
||||||
|
|
||||||
|
def test_enroll(self):
|
||||||
|
self.dev.open_sync()
|
||||||
|
|
||||||
|
# Must return a print
|
||||||
|
assert isinstance(self.enroll(), FPrint.Print)
|
||||||
|
|
||||||
|
self.dev.close_sync()
|
||||||
|
|
||||||
|
|
||||||
|
def test_verify(self):
|
||||||
|
self.dev.open_sync()
|
||||||
|
|
||||||
|
# Enroll a new print (will be the last), check that it verifies
|
||||||
|
p = self.enroll()
|
||||||
|
match, dev_print = self.dev.verify_sync(p)
|
||||||
|
assert match
|
||||||
|
|
||||||
|
# The returned "device" print is identical
|
||||||
|
assert p.equal(dev_print)
|
||||||
|
|
||||||
|
# We can do the same with it
|
||||||
|
match, dev_print2 = self.dev.verify_sync(dev_print)
|
||||||
|
assert match
|
||||||
|
|
||||||
|
# Now, enroll a new print, causing the old one to not match anymore
|
||||||
|
# (the device always claims to see the last enrolled print).
|
||||||
|
self.enroll()
|
||||||
|
match, dev_print = self.dev.verify_sync(p)
|
||||||
|
assert match is False
|
||||||
|
|
||||||
|
self.dev.close_sync()
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
try:
|
||||||
|
gi.require_version('FPrint', '2.0')
|
||||||
|
from gi.repository import FPrint
|
||||||
|
except Exception as e:
|
||||||
|
print("Missing dependencies: %s" % str(e))
|
||||||
|
sys.exit(77)
|
||||||
|
|
||||||
|
# avoid writing to stderr
|
||||||
|
unittest.main(testRunner=unittest.TextTestRunner(stream=sys.stdout, verbosity=2))
|
Loading…
Reference in a new issue