summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMarco Trevisan (Treviño) <mail@3v1n0.net>2020-09-24 05:45:35 +0200
committerMarco Trevisan (Treviño) <mail@3v1n0.net>2020-09-24 05:45:35 +0200
commit52410c3310ca41e45de68b5bb1393398b38d0850 (patch)
treea70c094e8bb6c2a05957348987500836309f71ed
parent0e123d0752538d834ee2cca8b471373369ad5e89 (diff)
parent174aa2c0917be87014849e4b72341f825f3de327 (diff)
Merge tag 'v1.90.3' into todv1.90.3+tod1
Tag 1.90.3 Git-EVTag-v0-SHA512: 67e0d995146cb82107480520589b6c90583602b3945e03f71c2fdb5f0f2c681143c67e824e0572ae93995e75112c402efaad101f01be5e86a75389e4f1421821
-rw-r--r--.gitlab-ci.yml73
-rw-r--r--NEWS19
-rw-r--r--demo/gtk-libfprint-test.c2
-rw-r--r--demo/org.freedesktop.libfprint.Demo.json2
-rw-r--r--examples/enroll.c2
-rw-r--r--examples/img-capture.c192
-rw-r--r--examples/meson.build2
-rw-r--r--examples/storage.c2
-rw-r--r--examples/storage.h2
-rw-r--r--examples/verify.c4
-rw-r--r--libfprint/drivers/aes3k.c32
-rw-r--r--libfprint/drivers/goodixmoc/goodix.c1453
-rw-r--r--libfprint/drivers/goodixmoc/goodix.h58
-rw-r--r--libfprint/drivers/goodixmoc/goodix_proto.c422
-rw-r--r--libfprint/drivers/goodixmoc/goodix_proto.h232
-rw-r--r--libfprint/drivers/synaptics/synaptics.c62
-rw-r--r--libfprint/fp-context.c8
-rw-r--r--libfprint/fp-device.h2
-rw-r--r--libfprint/fp-image-device.h2
-rw-r--r--libfprint/fpi-device.c17
-rw-r--r--libfprint/fpi-image-device.c8
-rw-r--r--libfprint/fpi-image-device.h5
-rw-r--r--libfprint/fpi-print.c2
-rw-r--r--libfprint/fpi-usb-transfer.c1
-rw-r--r--libfprint/fpi-usb-transfer.h8
-rw-r--r--libfprint/meson.build6
-rw-r--r--meson.build3
-rw-r--r--tests/goodixmoc/custom.ioctl307
-rwxr-xr-xtests/goodixmoc/custom.py50
-rw-r--r--tests/goodixmoc/device171
-rw-r--r--tests/meson.build1
31 files changed, 3033 insertions, 117 deletions
diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml
index 3ad55fb..3f0925b 100644
--- a/.gitlab-ci.yml
+++ b/.gitlab-ci.yml
@@ -3,6 +3,7 @@ include:
- project: 'wayland/ci-templates'
ref: master
file: '/templates/fedora.yml'
+ - remote: 'https://gitlab.gnome.org/GNOME/citemplates/-/raw/master/flatpak/flatpak_ci_initiative.yml'
variables:
extends: .libfprint_common_variables
@@ -108,56 +109,28 @@ test_indent:
- git diff
- "! git status -s | grep -q ."
-.flatpak_script_template: &flatpak_script
- script:
- - flatpak-builder --stop-at=${FLATPAK_MODULE} app ${MANIFEST_PATH}
- # Make sure to keep this in sync with the Flatpak manifest, all arguments
- # are passed except the config-args because we build it ourselves
- - flatpak build app meson --prefix=/app --libdir=lib ${MESON_ARGS} _build
- - flatpak build app ninja -C _build install
- - flatpak build app rm -rf /app/include/ /app/lib/pkgconfig/
- - flatpak-builder --finish-only --repo=repo app ${MANIFEST_PATH}
- # Generate a Flatpak bundle
- - flatpak build-bundle repo ${BUNDLE} --runtime-repo=${RUNTIME_REPO} ${DBUS_ID}
-
-.flatpak_artifacts_template: &flatpak_artifacts
- artifacts:
- paths:
- - ${BUNDLE}
- when: always
- expire_in: 30 days
-
-.flatpak_template: &flatpak
- <<: *flatpak_script
- <<: *flatpak_artifacts
-
-.flatpak_master_template: &flatpak_master
- image: registry.gitlab.gnome.org/gnome/gnome-runtime-images/gnome:3.32
- stage: flatpak
- variables:
- MANIFEST_PATH: "demo/org.freedesktop.libfprint.Demo.json"
- # From demo/org.freedesktop.libfprint.Demo.json
- MESON_ARGS: "-Dudev_rules=false -Dx11-examples=false -Dgtk-examples=true"
- FLATPAK_MODULE: "libfprint"
- DBUS_ID: "org.freedesktop.libfprint.Demo"
- <<: *flatpak
-
-flatpak-auto master:
- <<: *flatpak_master
- when: always
- only:
- - tags
- - master
-
-flatpak-manual master:
- <<: *flatpak_master
- when: manual
- except:
- refs:
- - tags
- - master
- variables:
- - $CI_PIPELINE_SOURCE == "schedule"
+flatpak:
+ stage: flatpak
+ extends: .flatpak
+ image: registry.gitlab.gnome.org/gnome/gnome-runtime-images/gnome:3.36
+ variables:
+ MANIFEST_PATH: "demo/org.freedesktop.libfprint.Demo.json"
+ FLATPAK_MODULE: "libfprint"
+ APP_ID: "org.freedesktop.libfprint.Demo"
+ rules:
+ - if: '$CI_PROJECT_PATH != "libfprint/libfprint"'
+ when: never
+ - if: '$CI_PIPELINE_SOURCE == "schedule"'
+ when: never
+ - if: '$CI_COMMIT_BRANCH == "master"'
+ when: always
+ - if: '$CI_COMMIT_TAG'
+ when: always
+ # For any other (commit), allow manual run.
+ # This excludes MRs which would create a duplicate pipeline
+ - if: '$CI_COMMIT_BRANCH'
+ when: manual
+ allow_failure: true
# CONTAINERS creation stage
container_fedora_build:
diff --git a/NEWS b/NEWS
index e7568d6..b01e023 100644
--- a/NEWS
+++ b/NEWS
@@ -1,6 +1,25 @@
This file lists notable changes in each release. For the full history of all
changes, see ChangeLog.
+2020-06-08: v1.90.3 release
+
+This release mostly contains support for a number of new match-on-chip
+devices. Most notable is the addition of the new goodixmoc driver.
+Currently the driver has the small caveat that we have no strategy to
+garbage collect old prints yet (a simple strategy could be implemented
+in fprintd).
+
+Highlights:
+ * New goodixmoc driver supporting Goodix USB devices:
+ 27C6:5840
+ 27C6:6496
+ 27C6:60A2
+ * Newly added support for Synaptics device:
+ 06CB:00E9
+ 06CB:00DF
+ * Fixed an issue with Synaptics devices sometimes not working at boot
+ * Fix issue with aes3k driver (#306)
+
2020-06-08: v1.90.2 release
This release contains a large amount of bug and regression fixes. These
diff --git a/demo/gtk-libfprint-test.c b/demo/gtk-libfprint-test.c
index 8026815..9ef8768 100644
--- a/demo/gtk-libfprint-test.c
+++ b/demo/gtk-libfprint-test.c
@@ -241,6 +241,8 @@ dev_capture_start_cb (FpDevice *dev,
if (error->domain == FP_DEVICE_RETRY ||
g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
libfprint_demo_set_mode (win, RETRY_MODE);
+ else if (g_error_matches (error, FP_DEVICE_ERROR, FP_DEVICE_ERROR_NOT_SUPPORTED))
+ libfprint_demo_set_mode (win, NOIMAGING_MODE);
else
libfprint_demo_set_mode (win, ERROR_MODE);
return;
diff --git a/demo/org.freedesktop.libfprint.Demo.json b/demo/org.freedesktop.libfprint.Demo.json
index 7429670..19516a4 100644
--- a/demo/org.freedesktop.libfprint.Demo.json
+++ b/demo/org.freedesktop.libfprint.Demo.json
@@ -1,7 +1,7 @@
{
"app-id": "org.freedesktop.libfprint.Demo",
"runtime": "org.gnome.Platform",
- "runtime-version": "3.32",
+ "runtime-version": "3.36",
"sdk": "org.gnome.Sdk",
"command": "gtk-libfprint-test",
"finish-args": [
diff --git a/examples/enroll.c b/examples/enroll.c
index ce6f755..17c231f 100644
--- a/examples/enroll.c
+++ b/examples/enroll.c
@@ -112,7 +112,7 @@ on_enroll_progress (FpDevice *device,
return;
}
- if (fp_device_supports_capture (device) &&
+ if (print && fp_print_get_image (print) &&
print_image_save (print, "enrolled.pgm"))
printf ("Wrote scanned image to enrolled.pgm\n");
diff --git a/examples/img-capture.c b/examples/img-capture.c
new file mode 100644
index 0000000..638aa7e
--- /dev/null
+++ b/examples/img-capture.c
@@ -0,0 +1,192 @@
+/*
+ * Example fingerprint verification program, which verifies the
+ * finger which has been previously enrolled to disk.
+ * Copyright (C) 2007 Daniel Drake <dsd@gentoo.org>
+ * Copyright (C) 2019 Marco Trevisan <marco.trevisan@canonical.com>
+ * Copyright (C) 2020 Vasily Khoruzhick <anarsoul@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 "example-capture"
+
+#include <stdio.h>
+#include <libfprint/fprint.h>
+#include <glib-unix.h>
+
+#include "storage.h"
+#include "utilities.h"
+
+typedef struct CaptureData
+{
+ GMainLoop *loop;
+ GCancellable *cancellable;
+ unsigned int sigint_handler;
+ int ret_value;
+ const char *filename;
+} CaptureData;
+
+static void
+capture_data_free (CaptureData *capture_data)
+{
+ g_clear_handle_id (&capture_data->sigint_handler, g_source_remove);
+ g_clear_object (&capture_data->cancellable);
+ g_main_loop_unref (capture_data->loop);
+ g_free (capture_data);
+}
+G_DEFINE_AUTOPTR_CLEANUP_FUNC (CaptureData, capture_data_free)
+
+static void
+on_device_closed (FpDevice *dev, GAsyncResult *res, void *user_data)
+{
+ CaptureData *capture_data = user_data;
+
+ g_autoptr(GError) error = NULL;
+
+ fp_device_close_finish (dev, res, &error);
+
+ if (error)
+ g_warning ("Failed closing device %s\n", error->message);
+
+ g_main_loop_quit (capture_data->loop);
+}
+
+static void
+capture_quit (FpDevice *dev,
+ CaptureData *capture_data)
+{
+ if (!fp_device_is_open (dev))
+ {
+ g_main_loop_quit (capture_data->loop);
+ return;
+ }
+
+ fp_device_close (dev, NULL, (GAsyncReadyCallback) on_device_closed, capture_data);
+}
+
+static void
+dev_capture_cb (FpDevice *dev,
+ GAsyncResult *res,
+ void *user_data)
+{
+ g_autoptr(GError) error = NULL;
+ CaptureData *capture_data = user_data;
+ FpImage *image = NULL;
+
+ g_clear_object (&capture_data->cancellable);
+
+ image = fp_device_capture_finish (dev, res, &error);
+ if (!image)
+ {
+ g_warning ("Error capturing data: %s", error->message);
+ capture_quit (dev, capture_data);
+ return;
+ }
+
+ save_image_to_pgm (image, capture_data->filename);
+
+ capture_quit (dev, capture_data);
+}
+
+static void
+start_capture (FpDevice *dev, CaptureData *capture_data)
+{
+ fp_device_capture (dev, TRUE, capture_data->cancellable, (GAsyncReadyCallback) dev_capture_cb, capture_data);
+}
+
+static void
+on_device_opened (FpDevice *dev, GAsyncResult *res, void *user_data)
+{
+ CaptureData *capture_data = user_data;
+
+ g_autoptr(GError) error = NULL;
+
+ if (!fp_device_open_finish (dev, res, &error))
+ {
+ g_warning ("Failed to open device: %s", error->message);
+ capture_quit (dev, capture_data);
+ return;
+ }
+
+ g_print ("Opened device. ");
+
+ start_capture (dev, capture_data);
+}
+
+static gboolean
+sigint_cb (void *user_data)
+{
+ CaptureData *capture_data = user_data;
+
+ g_cancellable_cancel (capture_data->cancellable);
+
+ return G_SOURCE_CONTINUE;
+}
+
+int
+main (int argc, const char *argv[])
+{
+ g_autoptr(FpContext) ctx = NULL;
+ g_autoptr(CaptureData) capture_data = NULL;
+ GPtrArray *devices;
+ FpDevice *dev;
+
+ setenv ("G_MESSAGES_DEBUG", "all", 0);
+ setenv ("LIBUSB_DEBUG", "3", 0);
+
+ ctx = fp_context_new ();
+
+ devices = fp_context_get_devices (ctx);
+ if (!devices)
+ {
+ g_warning ("Impossible to get devices");
+ return EXIT_FAILURE;
+ }
+
+ dev = discover_device (devices);
+ if (!dev)
+ {
+ g_warning ("No devices detected.");
+ return EXIT_FAILURE;
+ }
+
+ if (!fp_device_supports_capture (dev))
+ {
+ g_warning ("Device %s doesn't support capture",
+ fp_device_get_name (dev));
+ return EXIT_FAILURE;
+ }
+
+ capture_data = g_new0 (CaptureData, 1);
+ capture_data->ret_value = EXIT_FAILURE;
+ capture_data->loop = g_main_loop_new (NULL, FALSE);
+ capture_data->cancellable = g_cancellable_new ();
+ capture_data->sigint_handler = g_unix_signal_add_full (G_PRIORITY_HIGH,
+ SIGINT,
+ sigint_cb,
+ capture_data,
+ NULL);
+ if (argc == 2)
+ capture_data->filename = argv[1];
+ else
+ capture_data->filename = "finger.pgm";
+ fp_device_open (dev, capture_data->cancellable,
+ (GAsyncReadyCallback) on_device_opened,
+ capture_data);
+
+ g_main_loop_run (capture_data->loop);
+
+ return capture_data->ret_value;
+}
diff --git a/examples/meson.build b/examples/meson.build
index 90a1178..a0ea9ec 100644
--- a/examples/meson.build
+++ b/examples/meson.build
@@ -1,5 +1,5 @@
-examples = [ 'enroll', 'verify', 'manage-prints' ]
+examples = [ 'enroll', 'verify', 'manage-prints', 'img-capture' ]
foreach example: examples
executable(example,
[ example + '.c', 'storage.c', 'utilities.c' ],
diff --git a/examples/storage.c b/examples/storage.c
index ebffb33..4cccea2 100644
--- a/examples/storage.c
+++ b/examples/storage.c
@@ -180,7 +180,7 @@ print_create_template (FpDevice *dev, FpFinger finger)
}
-static gboolean
+gboolean
save_image_to_pgm (FpImage *img, const char *path)
{
FILE *fd = fopen (path, "w");
diff --git a/examples/storage.h b/examples/storage.h
index 6c6c220..0cc8269 100644
--- a/examples/storage.h
+++ b/examples/storage.h
@@ -28,3 +28,5 @@ FpPrint * print_create_template (FpDevice *dev,
FpFinger finger);
gboolean print_image_save (FpPrint *print,
const char *path);
+gboolean save_image_to_pgm (FpImage *img,
+ const char *path);
diff --git a/examples/verify.c b/examples/verify.c
index 23bfe8c..b47e1db 100644
--- a/examples/verify.c
+++ b/examples/verify.c
@@ -124,7 +124,7 @@ on_match_cb (FpDevice *dev, FpPrint *match, FpPrint *print,
return;
}
- if (print && fp_device_supports_capture (dev) &&
+ if (print && fp_print_get_image (print) &&
print_image_save (print, "verify.pgm"))
g_print ("Print image saved as verify.pgm\n");
@@ -256,7 +256,7 @@ start_verification (FpDevice *dev, VerifyData *verify_data)
g_print ("Print loaded. Time to verify!\n");
fp_device_verify (dev, verify_print, verify_data->cancellable,
- NULL, NULL, NULL,
+ on_match_cb, verify_data, NULL,
(GAsyncReadyCallback) on_verify_completed,
verify_data);
}
diff --git a/libfprint/drivers/aes3k.c b/libfprint/drivers/aes3k.c
index da3b6a3..db0d370 100644
--- a/libfprint/drivers/aes3k.c
+++ b/libfprint/drivers/aes3k.c
@@ -42,8 +42,7 @@
typedef struct
{
- FpiUsbTransfer *img_trf;
- gboolean deactivating;
+ GCancellable *img_trf_cancel;
} FpiDeviceAes3kPrivate;
#define CTRL_TIMEOUT 1000
@@ -77,25 +76,21 @@ img_cb (FpiUsbTransfer *transfer, FpDevice *device,
{
FpImageDevice *dev = FP_IMAGE_DEVICE (device);
FpiDeviceAes3k *self = FPI_DEVICE_AES3K (device);
- FpiDeviceAes3kPrivate *priv = fpi_device_aes3k_get_instance_private (self);
FpiDeviceAes3kClass *cls = FPI_DEVICE_AES3K_GET_CLASS (self);
unsigned char *ptr = transfer->buffer;
FpImage *tmp;
FpImage *img;
int i;
- priv->img_trf = NULL;
-
if (error)
{
if (g_error_matches (error,
G_IO_ERROR,
G_IO_ERROR_CANCELLED))
{
- /* Deactivation was completed. */
+ /* Cancellation implies we are deactivating. */
g_error_free (error);
- if (priv->deactivating)
- fpi_image_device_deactivate_complete (dev, NULL);
+ fpi_image_device_deactivate_complete (dev, NULL);
return;
}
@@ -126,21 +121,23 @@ img_cb (FpiUsbTransfer *transfer, FpDevice *device,
* it really has, then restart the capture */
fpi_image_device_report_finger_status (dev, FALSE);
+ /* Note: We always restart the transfer, it may already be cancelled though. */
do_capture (dev);
}
static void
do_capture (FpImageDevice *dev)
{
+ g_autoptr(FpiUsbTransfer) img_trf = NULL;
FpiDeviceAes3k *self = FPI_DEVICE_AES3K (dev);
FpiDeviceAes3kPrivate *priv = fpi_device_aes3k_get_instance_private (self);
FpiDeviceAes3kClass *cls = FPI_DEVICE_AES3K_GET_CLASS (self);
- priv->img_trf = fpi_usb_transfer_new (FP_DEVICE (dev));
- fpi_usb_transfer_fill_bulk (priv->img_trf, EP_IN, cls->data_buflen);
- priv->img_trf->short_is_error = TRUE;
- fpi_usb_transfer_submit (priv->img_trf, 0,
- fpi_device_get_cancellable (FP_DEVICE (dev)),
+ img_trf = fpi_usb_transfer_new (FP_DEVICE (dev));
+ fpi_usb_transfer_fill_bulk (img_trf, EP_IN, cls->data_buflen);
+ img_trf->short_is_error = TRUE;
+ fpi_usb_transfer_submit (g_steal_pointer (&img_trf), 0,
+ priv->img_trf_cancel,
img_cb, NULL);
}
@@ -159,7 +156,8 @@ aes3k_dev_activate (FpImageDevice *dev)
FpiDeviceAes3kPrivate *priv = fpi_device_aes3k_get_instance_private (self);
FpiDeviceAes3kClass *cls = FPI_DEVICE_AES3K_GET_CLASS (self);
- priv->deactivating = FALSE;
+ g_assert (!priv->img_trf_cancel);
+ priv->img_trf_cancel = g_cancellable_new ();
aes_write_regv (dev, cls->init_reqs, cls->init_reqs_len, init_reqs_cb, NULL);
}
@@ -169,10 +167,8 @@ aes3k_dev_deactivate (FpImageDevice *dev)
FpiDeviceAes3k *self = FPI_DEVICE_AES3K (dev);
FpiDeviceAes3kPrivate *priv = fpi_device_aes3k_get_instance_private (self);
- priv->deactivating = TRUE;
- if (priv->img_trf)
- return;
- fpi_image_device_deactivate_complete (dev, NULL);
+ /* Deactivation always finishes from the cancellation handler */
+ g_cancellable_cancel (priv->img_trf_cancel);
}
static void
diff --git a/libfprint/drivers/goodixmoc/goodix.c b/libfprint/drivers/goodixmoc/goodix.c
new file mode 100644
index 0000000..6a51cb8
--- /dev/null
+++ b/libfprint/drivers/goodixmoc/goodix.c
@@ -0,0 +1,1453 @@
+/*
+ * Goodix Moc driver for libfprint
+ * Copyright (C) 2019 Shenzhen Goodix Technology Co., Ltd.
+ *
+ * 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 "goodixmoc"
+
+#include "drivers_api.h"
+
+#include "goodix_proto.h"
+#include "goodix.h"
+
+
+/* Default enroll stages number */
+#define DEFAULT_ENROLL_SAMPLES 8
+/* Usb port setting */
+#define EP_IN (3 | FPI_USB_ENDPOINT_IN)
+#define EP_OUT (1 | FPI_USB_ENDPOINT_OUT)
+
+#define EP_IN_MAX_BUF_SIZE (2048)
+
+#define MAX_USER_ID_LEN (64)
+
+/* Command transfer timeout :ms*/
+#define CMD_TIMEOUT (1000)
+#define ACK_TIMEOUT (2000)
+#define DATA_TIMEOUT (5000)
+
+
+struct _FpiDeviceGoodixMoc
+{
+ FpDevice parent;
+ FpiSsm *task_ssm;
+ FpiSsm *cmd_ssm;
+ FpiUsbTransfer *cmd_transfer;
+ gboolean cmd_cancelable;
+ pgxfp_sensor_cfg_t sensorcfg;
+ gint enroll_stage;
+ gint max_enroll_stage;
+ GCancellable *cancellable;
+ GPtrArray *list_result;
+ guint8 template_id[TEMPLATE_ID_SIZE];
+ gboolean is_enroll_identify;
+
+};
+
+G_DEFINE_TYPE (FpiDeviceGoodixMoc, fpi_device_goodixmoc, FP_TYPE_DEVICE)
+
+typedef void (*SynCmdMsgCallback) (FpiDeviceGoodixMoc *self,
+ gxfp_cmd_response_t *resp,
+ GError *error);
+
+typedef struct
+{
+ guint8 cmd;
+ SynCmdMsgCallback callback;
+} CommandData;
+
+static gboolean parse_print_data (GVariant *data,
+ guint8 *finger,
+ const guint8 **tid,
+ gsize *tid_len,
+ const guint8 **user_id,
+ gsize *user_id_len);
+/******************************************************************************
+ *
+ * fp_cmd_xxx Function
+ *
+ *****************************************************************************/
+static void
+fp_cmd_receive_cb (FpiUsbTransfer *transfer,
+ FpDevice *device,
+ gpointer user_data,
+ GError *error)
+{
+ FpiDeviceGoodixMoc *self = FPI_DEVICE_GOODIXMOC (device);
+ CommandData *data = user_data;
+ int ret = -1, ssm_state = 0;
+ gxfp_cmd_response_t cmd_reponse = {0, };
+ pack_header header;
+ guint32 crc32_calc = 0;
+ guint16 cmd = 0;
+
+ if (error)
+ {
+ fpi_ssm_mark_failed (transfer->ssm, error);
+ return;
+ }
+ if (data == NULL)
+ {
+ fpi_ssm_mark_failed (transfer->ssm,
+ fpi_device_error_new (FP_DEVICE_ERROR_GENERAL));
+ return;
+ }
+ ssm_state = fpi_ssm_get_cur_state (transfer->ssm);
+ /* skip zero length package */
+ if (transfer->actual_length == 0)
+ {
+ fpi_ssm_jump_to_state (transfer->ssm, ssm_state);
+ return;
+ }
+
+ ret = gx_proto_parse_header (transfer->buffer, transfer->actual_length, &header);
+ if (ret != 0)
+ {
+ fpi_ssm_mark_failed (transfer->ssm,
+ fpi_device_error_new_msg (FP_DEVICE_ERROR_PROTO,
+ "Corrupted message received"));
+ return;
+ }
+
+ gx_proto_crc32_calc (transfer->buffer, PACKAGE_HEADER_SIZE + header.len, (uint8_t *) &crc32_calc);
+ if(crc32_calc != *(uint32_t *) (transfer->buffer + PACKAGE_HEADER_SIZE + header.len))
+ {
+ fpi_ssm_mark_failed (transfer->ssm,
+ fpi_device_error_new_msg (FP_DEVICE_ERROR_PROTO,
+ "Package crc check failed"));
+ return;
+ }
+
+ cmd = MAKE_CMD_EX (header.cmd0, header.cmd1);
+
+ ret = gx_proto_parse_body (cmd, &transfer->buffer[PACKAGE_HEADER_SIZE], header.len, &cmd_reponse);
+ if (ret != 0)
+ {
+ fpi_ssm_mark_failed (transfer->ssm,
+ fpi_device_error_new_msg (FP_DEVICE_ERROR_PROTO,
+ "Corrupted message received"));
+ return;
+ }
+ /* ack */
+ if(header.cmd0 == RESPONSE_PACKAGE_CMD)
+ {
+ if (data->cmd != cmd_reponse.parse_msg.ack_cmd)
+ {
+ fpi_ssm_mark_failed (transfer->ssm,
+ fpi_device_error_new_msg (FP_DEVICE_ERROR_PROTO,
+ "Unexpected response, got 0x%x",
+ cmd_reponse.parse_msg.ack_cmd));
+
+ return;
+ }
+ fpi_ssm_next_state (transfer->ssm);
+ return;
+ }
+ /* data */
+ if (data->cmd != header.cmd0)
+ {
+ fpi_ssm_mark_failed (transfer->ssm,
+ fpi_device_error_new_msg (FP_DEVICE_ERROR_PROTO,
+ "Unexpected cmd, got 0x%x",
+ header.cmd0));
+ return;
+ }
+ if (data->callback)
+ data->callback (self, &cmd_reponse, NULL);
+
+ fpi_ssm_mark_completed (transfer->ssm);
+}
+
+
+static void
+fp_cmd_run_state (FpiSsm *ssm,
+ FpDevice *dev)
+{
+ FpiUsbTransfer *transfer;
+ FpiDeviceGoodixMoc *self = FPI_DEVICE_GOODIXMOC (dev);
+
+ switch (fpi_ssm_get_cur_state (ssm))
+ {
+ case FP_CMD_SEND:
+ if (self->cmd_transfer)
+ {
+ self->cmd_transfer->ssm = ssm;
+ fpi_usb_transfer_submit (g_steal_pointer (&self->cmd_transfer),
+ CMD_TIMEOUT,
+ NULL,
+ fpi_ssm_usb_transfer_cb,
+ NULL);
+ }
+ else
+ {
+ fpi_ssm_next_state (ssm);
+ }
+ break;
+
+ case FP_CMD_GET_ACK:
+ transfer = fpi_usb_transfer_new (dev);
+ transfer->ssm = ssm;
+ fpi_usb_transfer_fill_bulk (transfer, EP_IN, EP_IN_MAX_BUF_SIZE);
+ fpi_usb_transfer_submit (transfer,
+ ACK_TIMEOUT,
+ NULL,
+ fp_cmd_receive_cb,
+ fpi_ssm_get_data (ssm));
+
+ break;
+
+ case FP_CMD_GET_DATA:
+ transfer = fpi_usb_transfer_new (dev);
+ transfer->ssm = ssm;
+ fpi_usb_transfer_fill_bulk (transfer, EP_IN, EP_IN_MAX_BUF_SIZE);
+ fpi_usb_transfer_submit (transfer,
+ self->cmd_cancelable ? 0 : DATA_TIMEOUT,
+ self->cmd_cancelable ? self->cancellable : NULL,
+ fp_cmd_receive_cb,
+ fpi_ssm_get_data (ssm));
+ break;
+ }
+
+}
+
+
+static void
+fp_cmd_ssm_done (FpiSsm *ssm, FpDevice *dev, GError *error)
+{
+ FpiDeviceGoodixMoc *self = FPI_DEVICE_GOODIXMOC (dev);
+ CommandData *data = fpi_ssm_get_data (ssm);
+
+ self->cmd_ssm = NULL;
+ /* Notify about the SSM failure from here instead. */
+ if (error)
+ {
+ if (data->callback)
+ data->callback (self, NULL, error);
+ else
+ g_error_free (error);
+ }
+}
+
+static FpiUsbTransfer *
+alloc_cmd_transfer (FpDevice *dev,
+ guint8 cmd0,
+ guint8 cmd1,
+ const guint8 *data,
+ guint16 data_len)
+{
+ gint ret = -1;
+
+ g_autoptr(FpiUsbTransfer) transfer = NULL;
+
+ guint32 total_len = data_len + PACKAGE_HEADER_SIZE + PACKAGE_CRC_SIZE;
+
+ g_return_val_if_fail (data || data_len == 0, NULL);
+
+ transfer = fpi_usb_transfer_new (dev);
+
+ fpi_usb_transfer_fill_bulk (transfer, EP_OUT, total_len);
+
+ ret = gx_proto_build_package (transfer->buffer, &total_len, MAKE_CMD_EX (cmd0, cmd1), data, data_len);
+
+ g_return_val_if_fail (ret == 0, NULL);
+
+ return g_steal_pointer (&transfer);
+}
+
+static void
+fp_cmd_ssm_done_data_free (CommandData *data)
+{
+ g_free (data);
+}
+
+static void
+goodix_sensor_cmd (FpiDeviceGoodixMoc *self,
+ guint8 cmd0,
+ guint8 cmd1,
+ gboolean bwait_data_delay,
+ const guint8 * payload,
+ gssize payload_len,
+ SynCmdMsgCallback callback)
+{
+
+ g_autoptr(FpiUsbTransfer) transfer = NULL;
+
+ CommandData *data = g_new0 (CommandData, 1);
+
+ transfer = alloc_cmd_transfer (FP_DEVICE (self), cmd0, cmd1, payload, payload_len);
+
+ data->cmd = cmd0;
+ data->callback = callback;
+
+ self->cmd_transfer = g_steal_pointer (&transfer);
+ self->cmd_cancelable = bwait_data_delay;
+
+ self->cmd_ssm = fpi_ssm_new (FP_DEVICE (self),
+ fp_cmd_run_state,
+ FP_CMD_NUM_STATES);
+
+ fpi_ssm_set_data (self->cmd_ssm, data, (GDestroyNotify) fp_cmd_ssm_done_data_free);
+
+ fpi_ssm_start (self->cmd_ssm, fp_cmd_ssm_done);
+
+
+}
+/******************************************************************************
+ *
+ * fp_verify_xxxx Function
+ *
+ *****************************************************************************/
+static void
+fp_verify_capture_cb (FpiDeviceGoodixMoc *self,
+ gxfp_cmd_response_t *resp,
+ GError *error)
+{
+ if (error)
+ {
+ fpi_ssm_mark_failed (self->task_ssm, error);
+ return;
+ }
+ if (resp->result >= GX_FAILED)
+ {
+ fp_dbg ("Capture sample failed, result: 0x%x", resp->result);
+ fpi_ssm_mark_failed (self->task_ssm,
+ fpi_device_retry_new (FP_DEVICE_RETRY_GENERAL));
+ return;
+ }
+
+ if (resp->capture_data_resp.img_quality == 0)
+ {
+ fpi_ssm_mark_failed (self->task_ssm,
+ fpi_device_retry_new (FP_DEVICE_RETRY_REMOVE_FINGER));
+ return;
+ }
+ else if (resp->capture_data_resp.img_coverage < 35)
+ {
+ fpi_ssm_mark_failed (self->task_ssm,
+ fpi_device_retry_new (FP_DEVICE_RETRY_CENTER_FINGER));
+ }
+ fpi_ssm_next_state (self->task_ssm);
+}
+
+static void
+fp_verify_cb (FpiDeviceGoodixMoc *self,
+ gxfp_cmd_response_t *resp,
+ GError *error)
+{
+ FpDevice *device = FP_DEVICE (self);
+ FpPrint *print = NULL;
+ GPtrArray *templates = NULL;
+ gint cnt = 0;
+ gboolean find = false;
+
+ if (error)
+ {
+ fpi_ssm_mark_failed (self->task_ssm, error);
+ return;
+ }
+ if (resp->verify.match)
+ {
+ if (fpi_device_get_current_action (device) == FPI_DEVICE_ACTION_VERIFY)
+ {
+
+ templates = g_ptr_array_new_with_free_func (g_object_unref);
+ fpi_device_get_verify_data (device, &print);
+ g_ptr_array_add (templates, g_object_ref_sink (print));
+
+ }
+ else
+ {
+ fpi_device_get_identify_data (device, &templates);
+ }
+ for (cnt = 0; cnt < templates->len; cnt++)
+ {
+ g_autoptr(GVariant) data = NULL;
+ guint8 finger;
+ const guint8 *user_id;
+ gsize user_id_len = 0;
+ const guint8 *tid;
+ gsize tid_len = 0;
+ print = g_ptr_array_index (templates, cnt);
+ g_object_get (print, "fpi-data", &data, NULL);
+ if (!parse_print_data (data, &finger, &tid, &tid_len, &user_id, &user_id_len))
+ {
+ fpi_ssm_mark_failed (self->task_ssm,
+ fpi_device_error_new_msg (FP_DEVICE_ERROR_DATA_INVALID,
+ "Parse print error"));
+ return;
+ }
+ if (memcmp (&resp->verify.template.tid, tid, TEMPLATE_ID_SIZE) == 0)
+ {
+ find = true;
+ break;
+ }
+
+ }
+ if (find)
+ {
+ if (fpi_device_get_current_action (device) == FPI_DEVICE_ACTION_VERIFY)
+ fpi_device_verify_report (device, FPI_MATCH_SUCCESS, NULL, error);
+ else
+ fpi_device_identify_report (device, print, print, error);
+ }
+ }
+
+ if (!find)
+ {
+ if (fpi_device_get_current_action (device) == FPI_DEVICE_ACTION_VERIFY)
+ fpi_device_verify_report (device, FPI_MATCH_FAIL, NULL, error);
+ else
+ fpi_device_identify_report (device, NULL, NULL, error);
+ }
+
+ fpi_ssm_mark_completed (self->task_ssm);
+
+}
+
+static void
+fp_verify_sm_run_state (FpiSsm *ssm, FpDevice *device)
+{
+ FpiDeviceGoodixMoc *self = FPI_DEVICE_GOODIXMOC (device);
+ guint8 param[3] = { 0 };
+ guint8 nonce[TEMPLATE_ID_SIZE] = { 0 };
+
+ param[0] = 0x01;
+ param[1] = self->sensorcfg->config[10];
+ param[2] = self->sensorcfg->config[11];
+
+ switch (fpi_ssm_get_cur_state (ssm))
+ {
+ case FP_VERIFY_CAPTURE:
+ goodix_sensor_cmd (self, MOC_CMD0_CAPTURE_DATA, MOC_CMD1_DEFAULT,
+ true,
+ (const guint8 *) &param,
+ G_N_ELEMENTS (param),
+ fp_verify_capture_cb);
+ break;
+
+ case FP_VERIFY_IDENTIFY:
+ goodix_sensor_cmd (self, MOC_CMD0_IDENTIFY, MOC_CMD1_DEFAULT,
+ false,
+ (const guint8 *) nonce,
+ TEMPLATE_ID_SIZE,
+ fp_verify_cb);
+ break;
+ }
+
+}
+
+static void
+fp_verify_ssm_done (FpiSsm *ssm, FpDevice *dev, GError *error)
+{
+ FpiDeviceGoodixMoc *self = FPI_DEVICE_GOODIXMOC (dev);
+
+ fp_info ("Verify complete!");
+
+ if (error && error->domain == FP_DEVICE_RETRY)
+ {
+ if (fpi_device_get_current_action (dev) == FPI_DEVICE_ACTION_VERIFY)
+ fpi_device_verify_report (dev, FPI_MATCH_ERROR, NULL, error);
+ else
+ fpi_device_identify_report (dev, NULL, NULL, error);
+ }
+
+ if (fpi_device_get_current_action (dev) == FPI_DEVICE_ACTION_VERIFY)
+ fpi_device_verify_complete (dev, error);
+
+ else
+ fpi_device_identify_complete (dev, error);
+
+ self->task_ssm = NULL;
+}
+
+
+/******************************************************************************
+ *
+ * fp__xxxx Function
+ *
+ *****************************************************************************/
+static gboolean
+encode_finger_id (
+ const guint8 * tid,
+ guint16 tid_len,
+ const guint8 * uid,
+ guint16 uid_len,
+ guint8 ** fid,
+ guint16 * fid_len
+ )
+{
+ guint8 * buffer = NULL;
+ guint16 offset = 0;
+
+ g_return_val_if_fail (tid != NULL, FALSE);
+ g_return_val_if_fail (uid != NULL, FALSE);
+ g_return_val_if_fail (fid != NULL, FALSE);
+ g_return_val_if_fail (fid_len != NULL, FALSE);
+
+ *fid_len = (guint16) (70 + uid_len); // must include fingerid length
+
+ *fid = (guint8 *) g_malloc0 (*fid_len + 2);
+
+ buffer = *fid;
+ offset = 0;
+ buffer[offset++] = LOBYTE (*fid_len);
+ buffer[offset++] = HIBYTE (*fid_len);
+
+ buffer[offset++] = 67;
+ buffer[offset++] = 1;
+ buffer[offset++] = 1; // finger index
+ buffer[offset++] = 0; //
+
+ offset += 32;
+
+ memcpy (&buffer[offset], tid, MIN (tid_len, TEMPLATE_ID_SIZE));
+ offset += 32; // offset == 68
+
+ buffer[offset++] = uid_len;
+ memcpy (&buffer[offset], uid, uid_len);
+ offset += (guint8) uid_len;
+
+ buffer[offset++] = 0;
+
+ if (offset != (*fid_len + 2))
+ {
+ memset (buffer, 0, *fid_len);
+ *fid_len = 0;
+
+ fp_err ("offset != fid_len, %d != %d", offset, *fid_len);
+ return FALSE;
+ }
+ *fid_len += 2;
+
+ return TRUE;
+}
+/******************************************************************************
+ *
+ * fp_enroll_xxxx Function
+ *
+ *****************************************************************************/
+
+static void
+fp_enroll_enum_cb (FpiDeviceGoodixMoc *self,
+ gxfp_cmd_response_t *resp,
+ GError *error)
+{
+ if (error)
+ {
+ fpi_ssm_mark_failed (self->task_ssm, error);
+ return;
+ }
+ if (resp->result != GX_SUCCESS)
+ {
+ fpi_ssm_mark_failed (self->task_ssm,
+ fpi_device_error_new_msg (FP_DEVICE_ERROR_GENERAL,
+ "Failed to enumerate fingers, result: 0x%x",
+ resp->result));
+ return;
+ }
+ fpi_ssm_jump_to_state (self->task_ssm, FP_ENROLL_CAPTURE);
+}
+
+static void
+fp_enroll_identify_cb (FpiDeviceGoodixMoc *self,
+ gxfp_cmd_response_t *resp,
+ GError *error)
+{
+ if (error)
+ {
+ fpi_ssm_mark_failed (self->task_ssm, error);
+ return;
+ }
+ if (resp->verify.match)
+ {
+ fpi_ssm_mark_failed (self->task_ssm,
+ fpi_device_error_new_msg (FP_DEVICE_ERROR_DATA_DUPLICATE,
+ "Finger is too similar to another, try use a different finger"));
+ // maybe need fpi_device_enroll_report_message ...
+ return;
+ }
+ fpi_ssm_next_state (self->task_ssm);
+
+}
+
+static void
+fp_enroll_init_cb (FpiDeviceGoodixMoc *self,
+ gxfp_cmd_response_t *resp,
+ GError *error)
+{
+ if (error)
+ {
+ fpi_ssm_mark_failed (self->task_ssm, error);
+ return;
+ }
+ memcpy (self->template_id, resp->enroll_init.tid, TEMPLATE_ID_SIZE);
+ fpi_ssm_next_state (self->task_ssm);
+}
+
+static void
+fp_enroll_capture_cb (FpiDeviceGoodixMoc *self,
+ gxfp_cmd_response_t *resp,
+ GError *error)
+{
+ if (error)
+ {
+ fpi_ssm_mark_failed (self->task_ssm, error);
+ return;
+ }
+ /* */
+ if (resp->result >= GX_FAILED)
+ {
+ fp_warn ("Capture sample failed, result: 0x%x", resp->result);
+ fpi_device_enroll_progress (FP_DEVICE (self),
+ self->enroll_stage,
+ NULL,
+ fpi_device_retry_new (FP_DEVICE_RETRY_GENERAL));
+ fpi_ssm_jump_to_state (self->task_ssm, FP_ENROLL_CAPTURE);
+ return;
+ }
+
+ if ((resp->capture_data_resp.img_quality < self->sensorcfg->config[4]) ||
+ (resp->capture_data_resp.img_coverage < self->sensorcfg->config[5]))
+ {
+ fp_warn ("Capture sample poor quality(%d): %d or coverage(%d): %d",
+ self->sensorcfg->config[4],
+ resp->capture_data_resp.img_quality,
+ self->sensorcfg->config[5],
+ resp->capture_data_resp.img_coverage);
+ fpi_device_enroll_progress (FP_DEVICE (self),
+ self->enroll_stage,
+ NULL,
+ fpi_device_retry_new (FP_DEVICE_RETRY_CENTER_FINGER));
+ fpi_ssm_jump_to_state (self->task_ssm, FP_ENROLL_CAPTURE);
+ return;
+ }
+ if (self->is_enroll_identify)
+ {
+ self->is_enroll_identify = false;
+ fpi_ssm_jump_to_state (self->task_ssm, FP_ENROLL_IDENTIFY);
+ }
+ else
+ {
+ fpi_ssm_next_state (self->task_ssm);
+ }
+
+}
+static void
+fp_enroll_update_cb (FpiDeviceGoodixMoc *self,
+ gxfp_cmd_response_t *resp,
+ GError *error)
+{
+ if (error)
+ {
+ fpi_ssm_mark_failed (self->task_ssm, error);
+ return;
+ }
+
+ if (resp->enroll_update.img_preoverlay > self->sensorcfg->config[3])
+ {
+ fp_dbg ("Sample overlapping ratio is too High(%d): %d ",
+ self->sensorcfg->config[3],
+ resp->enroll_update.img_preoverlay);
+ /* here should tips move finger and try again */
+ fpi_device_enroll_progress (FP_DEVICE (self),
+ self->enroll_stage,
+ NULL,
+ fpi_device_retry_new (FP_DEVICE_RETRY_REMOVE_FINGER));
+ }
+ else if (resp->enroll_update.rollback)
+ {
+ fpi_device_enroll_progress (FP_DEVICE (self),
+ self->enroll_stage,
+ NULL,
+ fpi_device_retry_new (FP_DEVICE_RETRY_GENERAL));
+ }
+ else
+ {
+ self->enroll_stage++;
+ fpi_device_enroll_progress (FP_DEVICE (self), self->enroll_stage, NULL, NULL);
+ }
+ /* if enroll complete, no need to wait finger up */
+ if (self->enroll_stage >= self->max_enroll_stage)
+ {
+ fpi_ssm_jump_to_state (self->task_ssm, FP_ENROLL_CHECK_DUPLICATE);
+ return;
+ }
+
+ fpi_ssm_next_state (self->task_ssm);
+}
+
+static void
+fp_enroll_check_duplicate_cb (FpiDeviceGoodixMoc *self,
+ gxfp_cmd_response_t *resp,
+ GError *error)
+{
+ if (error)
+ {
+ fpi_ssm_mark_failed (self->task_ssm, error);
+ return;
+ }
+ if (resp->check_duplicate_resp.duplicate)
+ {
+ fpi_ssm_mark_failed (self->task_ssm,
+ fpi_device_error_new_msg (FP_DEVICE_ERROR_DATA_DUPLICATE,
+ "Finger has already enrolled"));
+ return;
+ }
+
+ fpi_ssm_next_state (self->task_ssm);
+}
+
+static void
+fp_enroll_commit_cb (FpiDeviceGoodixMoc *self,
+ gxfp_cmd_response_t *resp,
+ GError *error)
+{
+ if (error)
+ {
+ fpi_ssm_mark_failed (self->task_ssm, error);
+ return;
+ }
+ if (resp->result >= GX_FAILED)
+ {
+ fpi_ssm_mark_failed (self->task_ssm,
+ fpi_device_error_new_msg (FP_DEVICE_ERROR_PROTO,
+ "Commit template failed with errcode: 0x%x", resp->result));
+ return;
+ }
+ fpi_ssm_next_state (self->task_ssm);
+}
+
+static void
+fp_finger_mode_cb (FpiDeviceGoodixMoc *self,
+ gxfp_cmd_response_t *resp,
+ GError *error)
+{
+ if (error)
+ {
+ fpi_ssm_mark_failed (self->task_ssm, error);
+ return;
+ }
+ /* if reach max timeout(5sec) finger not up, swtich to finger up again */
+ if (resp->finger_status.status == GX_ERROR_WAIT_FINGER_UP_TIMEOUT)
+ {
+ fpi_ssm_jump_to_state (self->task_ssm, FP_ENROLL_WAIT_FINGER_UP);
+ return;
+ }
+ else if (resp->finger_status.status != GX_SUCCESS)
+ {
+ fpi_ssm_mark_failed (self->task_ssm,
+ fpi_device_error_new_msg (FP_DEVICE_ERROR_PROTO,
+ "Switch finger mode failed"));
+ return;
+ }
+ if (self->enroll_stage < self->max_enroll_stage)
+ {
+ fpi_ssm_jump_to_state (self->task_ssm, FP_ENROLL_CAPTURE);
+ return;
+ }
+ fpi_ssm_next_state (self->task_ssm);
+}
+
+static void
+fp_enroll_sm_run_state (FpiSsm *ssm, FpDevice *device)
+{
+ FpiDeviceGoodixMoc *self = FPI_DEVICE_GOODIXMOC (device);
+ FpPrint *print = NULL;
+ GVariant *data = NULL;
+ GVariant *uid = NULL;
+ GVariant *tid = NULL;
+ guint finger;
+ guint16 user_id_len;
+ guint16 payload_len = 0;
+ g_autofree gchar *user_id = NULL;
+ g_autofree guint8 *payload = NULL;
+ guint8 dummy[3] = { 0 };
+
+ dummy[1] = self->sensorcfg->config[4];
+ dummy[2] = self->sensorcfg->config[5];
+
+ switch (fpi_ssm_get_cur_state (ssm))
+ {
+ case FP_ENROLL_ENUM:
+ {
+ goodix_sensor_cmd (self, MOC_CMD0_GETFINGERLIST, MOC_CMD1_DEFAULT,
+ false,
+ (const guint8 *) &dummy,
+ 1,
+ fp_enroll_enum_cb);
+ }
+ break;
+
+ case FP_ENROLL_IDENTIFY:
+ {
+ dummy[0] = 0x01;
+ dummy[1] = self->sensorcfg->config[10];
+ dummy[2] = self->sensorcfg->config[11];
+ goodix_sensor_cmd (self, MOC_CMD0_IDENTIFY, MOC_CMD1_DEFAULT,
+ false,
+ (const guint8 *) &self->template_id,
+ TEMPLATE_ID_SIZE,
+ fp_enroll_identify_cb);
+ }
+ break;
+
+ case FP_ENROLL_CREATE:
+ {
+ goodix_sensor_cmd (self, MOC_CMD0_ENROLL_INIT, MOC_CMD1_DEFAULT,
+ false,
+ (const guint8 *) &dummy,
+ 1,
+ fp_enroll_init_cb);
+ }
+ break;
+
+ case FP_ENROLL_CAPTURE:
+ goodix_sensor_cmd (self, MOC_CMD0_CAPTURE_DATA, MOC_CMD1_DEFAULT,
+ true,
+ (const guint8 *) &dummy,
+ 3,
+ fp_enroll_capture_cb);
+ break;
+
+ case FP_ENROLL_UPDATE:
+ dummy[0] = 1;
+ dummy[1] = self->sensorcfg->config[2];
+ dummy[2] = self->sensorcfg->config[3];
+ goodix_sensor_cmd (self, MOC_CMD0_ENROLL, MOC_CMD1_DEFAULT,
+ false,
+ (const guint8 *) &dummy,
+ 3,
+ fp_enroll_update_cb);
+ break;
+
+ case FP_ENROLL_WAIT_FINGER_UP:
+ dummy[0] = 0;
+ goodix_sensor_cmd (self, MOC_CMD0_FINGER_MODE, MOC_CMD1_SET_FINGER_UP,
+ true,
+ (const guint8 *) &dummy,
+ 1,
+ fp_finger_mode_cb);
+ break;
+
+ case FP_ENROLL_CHECK_DUPLICATE:
+ goodix_sensor_cmd (self, MOC_CMD0_CHECK4DUPLICATE, MOC_CMD1_DEFAULT,
+ false,
+ (const guint8 *) &dummy,
+ 3,
+ fp_enroll_check_duplicate_cb);
+ break;
+
+ case FP_ENROLL_COMMIT:
+ {
+ fpi_device_get_enroll_data (device, &print);
+ user_id = fpi_print_generate_user_id (print);
+ user_id_len = strlen (user_id);
+ user_id_len = MIN (100, user_id_len);
+ finger = 1;
+
+ if (g_strcmp0 (g_getenv ("FP_DEVICE_EMULATION"), "1") == 0)
+ memset (self->template_id, 0, TEMPLATE_ID_SIZE);
+ uid = g_variant_new_fixed_array (G_VARIANT_TYPE_BYTE,
+ user_id,
+ user_id_len,
+ 1);
+
+ tid = g_variant_new_fixed_array (G_VARIANT_TYPE_BYTE,
+ self->template_id,
+ TEMPLATE_ID_SIZE,
+ 1);
+
+ data = g_variant_new ("(y@ay@ay)",
+ finger,
+ tid,
+ uid);
+
+ fpi_print_set_type (print, FPI_PRINT_RAW);
+ fpi_print_set_device_stored (print, TRUE);
+ g_object_set (print, "fpi-data", data, NULL);
+ g_object_set (print, "description", user_id, NULL);
+
+ g_debug ("user_id: %s, user_id_len: %d, finger: %d", user_id, user_id_len, finger);
+
+ if(!encode_finger_id (self->template_id,
+ TEMPLATE_ID_SIZE,
+ (guint8 *) user_id,
+ user_id_len,
+ &payload,
+ &payload_len))
+ {
+ fpi_ssm_mark_failed (ssm,
+ fpi_device_error_new_msg (FP_DEVICE_ERROR_PROTO,
+ "encode_finger_id failed"));
+ return;
+ }
+ goodix_sensor_cmd (self, MOC_CMD0_COMMITENROLLMENT, MOC_CMD1_DEFAULT,
+ false,
+ (const guint8 *) payload,
+ payload_len,
+ fp_enroll_commit_cb);
+
+ }
+ break;
+ }
+
+
+}
+
+static void
+fp_enroll_ssm_done (FpiSsm *ssm, FpDevice *dev, GError *error)
+{
+ FpiDeviceGoodixMoc *self = FPI_DEVICE_GOODIXMOC (dev);
+ FpPrint *print = NULL;
+
+ if (error)
+ {
+ fpi_device_enroll_complete (dev, NULL, error);
+ return;
+ }
+ fp_info ("Enrollment complete!");
+
+ fpi_device_get_enroll_data (FP_DEVICE (self), &print);
+
+ fpi_device_enroll_complete (FP_DEVICE (self), g_object_ref (print), NULL);
+
+ self->task_ssm = NULL;
+}
+/******************************************************************************
+ *
+ * fp_init_xxxx Function
+ *
+ *****************************************************************************/
+static void
+fp_init_version_cb (FpiDeviceGoodixMoc *self,
+ gxfp_cmd_response_t *resp,
+ GError *error)
+{
+ g_autofree gchar *fw_type = NULL;
+ g_autofree gchar *fw_version = NULL;
+
+ if (error)
+ {
+ fpi_ssm_mark_failed (self->task_ssm, error);
+ return;
+ }
+
+ G_STATIC_ASSERT (sizeof (resp->version_info.fwtype) == 8);
+ G_STATIC_ASSERT (sizeof (resp->version_info.fwversion) == 8);
+
+ fw_type = g_strndup ((const char *) resp->version_info.fwtype, sizeof (resp->version_info.fwtype));
+
+ fp_info ("Firmware type: %s", fw_type);
+ if (g_strcmp0 (fw_type, "APP") != 0)
+ {
+ fpi_ssm_mark_failed (self->task_ssm,
+ fpi_device_error_new_msg (FP_DEVICE_ERROR_NOT_SUPPORTED,
+ "Please update firmware using fwupd"));
+ return;
+ }
+ fw_version = g_strndup ((const char *) resp->version_info.fwversion, sizeof (resp->version_info.fwversion));
+ fp_info ("Firmware version: %s", fw_version);
+ fpi_ssm_next_state (self->task_ssm);
+}
+
+static void
+fp_init_config_cb (FpiDeviceGoodixMoc *self,
+ gxfp_cmd_response_t *resp,
+ GError *error)
+{
+ if (error)
+ {
+ fpi_ssm_mark_failed (self->task_ssm, error);
+ return;
+ }
+
+ fpi_ssm_next_state (self->task_ssm);
+}
+
+
+static void
+fp_init_sm_run_state (FpiSsm *ssm, FpDevice *device)
+{
+ FpiDeviceGoodixMoc *self = FPI_DEVICE_GOODIXMOC (device);
+ guint8 dummy = 0;
+
+ switch (fpi_ssm_get_cur_state (ssm))
+ {
+ case FP_INIT_VERSION:
+ goodix_sensor_cmd (self, MOC_CMD0_GET_VERSION, MOC_CMD1_DEFAULT,
+ false,
+ &dummy,
+ 1,
+ fp_init_version_cb);
+ break;
+
+ case FP_INIT_CONFIG:
+ goodix_sensor_cmd (self, MOC_CMD0_UPDATE_CONFIG, MOC_CMD1_WRITE_CFG_TO_FLASH,
+ false,
+ (guint8 *) self->sensorcfg,
+ sizeof (gxfp_sensor_cfg_t),
+ fp_init_config_cb);
+ break;
+ }
+
+
+}
+
+static void
+fp_init_ssm_done (FpiSsm *ssm, FpDevice *dev, GError *error)
+{
+ FpiDeviceGoodixMoc *self = FPI_DEVICE_GOODIXMOC (dev);
+
+ if (error)
+ {
+ fpi_device_open_complete (dev, error);
+ return;
+ }
+ self->task_ssm = NULL;
+ fpi_device_open_complete (dev, NULL);
+}
+/******************************************************************************
+ *
+ * fp_template_delete Function
+ *
+ *****************************************************************************/
+static gboolean
+parse_print_data (GVariant *data,
+ guint8 *finger,
+ const guint8 **tid,
+ gsize *tid_len,
+ const guint8 **user_id,
+ gsize *user_id_len)
+{
+ g_autoptr(GVariant) user_id_var = NULL;
+ g_autoptr(GVariant) tid_var = NULL;
+
+ g_return_val_if_fail (data != NULL, FALSE);
+ g_return_val_if_fail (finger != NULL, FALSE);
+ g_return_val_if_fail (tid != NULL, FALSE);
+ g_return_val_if_fail (tid_len != NULL, FALSE);
+ g_return_val_if_fail (user_id != NULL, FALSE);
+ g_return_val_if_fail (user_id_len != NULL, FALSE);
+
+ *tid = NULL;
+ *tid_len = 0;
+ *user_id = NULL;
+ *user_id_len = 0;
+
+
+ if (!g_variant_check_format_string (data, "(y@ay@ay)", FALSE))
+ return FALSE;
+
+ g_variant_get (data,
+ "(y@ay@ay)",
+ finger,
+ &tid_var,
+ &user_id_var);
+
+ *tid = g_variant_get_fixed_array (tid_var, tid_len, 1);
+ *user_id = g_variant_get_fixed_array (user_id_var, user_id_len, 1);
+
+ if (*user_id_len == 0 || *user_id_len > 100)
+ return FALSE;
+
+ if (*user_id_len <= 0 || *user_id[0] == ' ')
+ return FALSE;
+
+ if(*tid_len != TEMPLATE_ID_SIZE)
+ return FALSE;
+
+ return TRUE;
+}
+
+static void
+fp_template_delete_cb (FpiDeviceGoodixMoc *self,
+ gxfp_cmd_response_t *resp,
+ GError *error)
+{
+ FpDevice *device = FP_DEVICE (self);
+
+ if (error)
+ {
+ fpi_device_delete_complete (device, error);
+ return;
+ }
+ if ((resp->result >= GX_FAILED) && (resp->result != GX_ERROR_FINGER_ID_NOEXIST))
+ {
+ fpi_device_delete_complete (FP_DEVICE (self),
+ fpi_device_error_new_msg (FP_DEVICE_ERROR_GENERAL,
+ "Failed delete enrolled users, result: 0x%x",
+ resp->result));
+ return;
+ }
+
+ fp_info ("Successfully deleted enrolled user");
+ fpi_device_delete_complete (device, NULL);
+}
+/******************************************************************************
+ *
+ * fp_template_list Function
+ *
+ *****************************************************************************/
+static void
+fp_template_list_cb (FpiDeviceGoodixMoc *self,
+ gxfp_cmd_response_t *resp,
+ GError *error)
+{
+ FpDevice *device = FP_DEVICE (self);
+
+ if (error)
+ {
+ fpi_device_list_complete (FP_DEVICE (self), NULL, error);
+ return;
+ }
+ if (resp->result != GX_SUCCESS)
+ {
+ fp_info ("Failed to query enrolled users: %d", resp->result);
+ fpi_device_list_complete (FP_DEVICE (self),
+ NULL,
+ fpi_device_error_new_msg (FP_DEVICE_ERROR_GENERAL,
+ "Failed to query enrolled users, result: 0x%x",
+ resp->result));
+ return;
+ }
+
+ self->list_result = g_ptr_array_new_with_free_func (g_object_unref);
+
+ if(resp->finger_list_resp.finger_num == 0)
+ {
+ fp_info ("Database is empty");
+ fpi_device_list_complete (device,
+ g_steal_pointer (&self->list_result),
+ NULL);
+ return;
+ }
+
+ for (int n = 0; n < resp->finger_list_resp.finger_num; n++)
+ {
+ GVariant *data = NULL;
+ GVariant *tid = NULL;
+ GVariant *uid = NULL;
+ FpPrint *print;
+ gchar *userid;
+
+ userid = (gchar *) resp->finger_list_resp.finger_list[n].payload.data;
+
+ print = fp_print_new (FP_DEVICE (self));
+
+ tid = g_variant_new_fixed_array (G_VARIANT_TYPE_BYTE,
+ resp->finger_list_resp.finger_list[n].tid,
+ TEMPLATE_ID_SIZE,
+ 1);
+
+ uid = g_variant_new_fixed_array (G_VARIANT_TYPE_BYTE,
+ resp->finger_list_resp.finger_list[n].payload.data,
+ resp->finger_list_resp.finger_list[n].payload.size,
+ 1);
+
+ data = g_variant_new ("(y@ay@ay)",
+ resp->finger_list_resp.finger_list[n].finger_index,
+ tid,
+ uid);
+
+ fpi_print_set_type (print, FPI_PRINT_RAW);
+ fpi_print_set_device_stored (print, TRUE);
+ g_object_set (print, "fpi-data", data, NULL);
+ g_object_set (print, "description", userid, NULL);
+ fpi_print_fill_from_user_id (print, userid);
+ g_ptr_array_add (self->list_result, g_object_ref_sink (print));
+ }
+
+ fp_info ("Query complete!");
+ fpi_device_list_complete (device,
+ g_steal_pointer (&self->list_result),
+ NULL);
+
+}
+
+/******************************************************************************
+ *
+ * Interface Function
+ *
+ *****************************************************************************/
+static void
+gx_fp_probe (FpDevice *device)
+{
+ GUsbDevice *usb_dev;
+ GError *error = NULL;
+ FpiDeviceGoodixMoc *self = FPI_DEVICE_GOODIXMOC (device);
+ g_autofree gchar *serial = NULL;
+ gint productid = 0;
+
+ /* Claim usb interface */
+ usb_dev = fpi_device_get_usb_device (device);
+ if (!g_usb_device_open (usb_dev, &error))
+ {
+ fpi_device_probe_complete (device, NULL, NULL, error);
+ return;
+ }
+
+ if (!g_usb_device_reset (usb_dev, &error))
+ goto err_close;
+
+ if (!g_usb_device_claim_interface (usb_dev, 0, 0, &error))
+ goto err_close;
+
+ if (g_strcmp0 (g_getenv ("FP_DEVICE_EMULATION"), "1") == 0)
+ {
+
+ serial = g_strdup ("emulated-device");
+ }
+ else
+ {
+ serial = g_usb_device_get_string_descriptor (usb_dev,
+ g_usb_device_get_serial_number_index (usb_dev),
+ &error);
+
+ if (serial && !g_str_has_suffix (serial, "B0"))
+ fp_warn ("Device with serial %s not supported", serial);
+ if (error)
+ {
+ g_usb_device_release_interface (fpi_device_get_usb_device (FP_DEVICE (device)),
+ 0, 0, NULL);
+ goto err_close;
+ }
+ }
+ productid = g_usb_device_get_pid (usb_dev);
+ switch (productid)
+ {
+ case 0x6496:
+ case 0x60A2:
+ self->max_enroll_stage = 12;
+ break;
+
+ default:
+ self->max_enroll_stage = DEFAULT_ENROLL_SAMPLES;
+ break;
+ }
+
+ fpi_device_set_nr_enroll_stages (device, self->max_enroll_stage);
+
+ g_usb_device_close (usb_dev, NULL);
+ fpi_device_probe_complete (device, serial, NULL, error);
+ return;
+err_close:
+
+ g_usb_device_close (usb_dev, NULL);
+ fpi_device_probe_complete (device, NULL, NULL, error);
+
+}
+
+static void
+gx_fp_init (FpDevice *device)
+{
+ FpiDeviceGoodixMoc *self = FPI_DEVICE_GOODIXMOC (device);
+ GError *error = NULL;
+ int ret = 0;
+
+ self->cancellable = g_cancellable_new ();
+
+ self->sensorcfg = g_new0 (gxfp_sensor_cfg_t, 1);
+
+ ret = gx_proto_init_sensor_config (self->sensorcfg);
+ if (ret != 0)
+ {
+ error = fpi_device_error_new_msg (FP_DEVICE_ERROR_GENERAL, "Init sensor failed");
+ fpi_device_open_complete (FP_DEVICE (self), error);
+ return;
+ }
+ self->sensorcfg->config[6] = self->max_enroll_stage;
+
+ if (!g_usb_device_reset (fpi_device_get_usb_device (device), &error))
+ {
+ fpi_device_open_complete (FP_DEVICE (self), error);
+ return;
+ }
+
+ /* Claim usb interface */
+ if (!g_usb_device_claim_interface (fpi_device_get_usb_device (device), 0, 0, &error))
+ {
+ fpi_device_open_complete (FP_DEVICE (self), error);
+ return;
+ }
+
+ self->task_ssm = fpi_ssm_new (device, fp_init_sm_run_state,
+ FP_INIT_NUM_STATES);
+
+ fpi_ssm_start (self->task_ssm, fp_init_ssm_done);
+
+}
+
+static void
+gx_fp_exit (FpDevice *device)
+{
+ FpiDeviceGoodixMoc *self = FPI_DEVICE_GOODIXMOC (device);
+ GError *error = NULL;
+
+ g_clear_object (&self->cancellable);
+ g_clear_pointer (&self->sensorcfg, g_free);
+
+ /* Release usb interface */
+ g_usb_device_release_interface (fpi_device_get_usb_device (FP_DEVICE (device)),
+ 0, 0, &error);
+
+ /* Notify close complete */
+ fpi_device_close_complete (FP_DEVICE (self), error);
+}
+
+
+static void
+gx_fp_verify_identify (FpDevice *device)
+{
+ FpiDeviceGoodixMoc *self = FPI_DEVICE_GOODIXMOC (device);
+
+ self->task_ssm = fpi_ssm_new (device, fp_verify_sm_run_state,
+ FP_VERIFY_NUM_STATES);
+
+ fpi_ssm_start (self->task_ssm, fp_verify_ssm_done);
+
+}
+
+static void
+gx_fp_enroll (FpDevice *device)
+{
+
+ FpiDeviceGoodixMoc *self = FPI_DEVICE_GOODIXMOC (device);
+
+ self->enroll_stage = 0;
+ self->is_enroll_identify = true;
+
+ self->task_ssm = fpi_ssm_new (device, fp_enroll_sm_run_state,
+ FP_ENROLL_NUM_STATES);
+
+ fpi_ssm_start (self->task_ssm, fp_enroll_ssm_done);
+
+}
+
+
+static void
+gx_fp_template_list (FpDevice *device)
+{
+
+ FpiDeviceGoodixMoc *self = FPI_DEVICE_GOODIXMOC (device);
+ guint8 dummy[1] = { 0 };
+
+ G_DEBUG_HERE ();
+
+ goodix_sensor_cmd (self, MOC_CMD0_GETFINGERLIST, MOC_CMD1_DEFAULT,
+ false,
+ (const guint8 *) &dummy,
+ 1,
+ fp_template_list_cb);
+}
+
+
+
+static void
+gx_fp_template_delete (FpDevice *device)
+{
+
+ FpiDeviceGoodixMoc *self = FPI_DEVICE_GOODIXMOC (device);
+ FpPrint *print = NULL;
+
+ g_autoptr(GVariant) data = NULL;
+ guint8 finger;
+ const guint8 *user_id;
+ gsize user_id_len = 0;
+ const guint8 *tid;
+ gsize tid_len = 0;
+ gsize payload_len = 0;
+ g_autofree guint8 *payload = NULL;
+
+ fpi_device_get_delete_data (device, &print);
+
+ g_object_get (print, "fpi-data", &data, NULL);
+
+ if (!parse_print_data (data, &finger, &tid, &tid_len, &user_id, &user_id_len))
+ {
+ fpi_device_delete_complete (device,
+ fpi_device_error_new (FP_DEVICE_ERROR_DATA_INVALID));
+ return;
+ }
+ if (!encode_finger_id (tid, tid_len, user_id, user_id_len, &payload, (guint16 *) &payload_len))
+ {
+ fpi_device_delete_complete (device,
+ fpi_device_error_new_msg (FP_DEVICE_ERROR_PROTO,
+ "encode_finger_id failed"));
+ return;
+ }
+
+ goodix_sensor_cmd (self, MOC_CMD0_DELETETEMPLATE, MOC_CMD1_DEFAULT,
+ false,
+ (const guint8 *) payload,
+ payload_len,
+ fp_template_delete_cb);
+
+}
+
+static void
+fpi_device_goodixmoc_init (FpiDeviceGoodixMoc *self)
+{
+
+}
+
+static void
+gx_fp_cancel (FpDevice *device)
+{
+
+ FpiDeviceGoodixMoc *self = FPI_DEVICE_GOODIXMOC (device);
+
+ /* Cancel any current interrupt transfer (resulting us to go into
+ * response reading mode again); then create a new cancellable
+ * for the next transfers. */
+ g_cancellable_cancel (self->cancellable);
+ g_clear_object (&self->cancellable);
+ self->cancellable = g_cancellable_new ();
+
+}
+
+static const FpIdEntry id_table[] = {
+ { .vid = 0x27c6, .pid = 0x5840, },
+ { .vid = 0x27c6, .pid = 0x6496, },
+ { .vid = 0x27c6, .pid = 0x60A2, },
+ { .vid = 0, .pid = 0, .driver_data = 0 }, /* terminating entry */
+};
+
+
+static void
+fpi_device_goodixmoc_class_init (FpiDeviceGoodixMocClass *klass)
+{
+ FpDeviceClass *dev_class = FP_DEVICE_CLASS (klass);
+
+ dev_class->id = "goodixmoc";
+ dev_class->full_name = "Goodix MOC Fingerprint Sensor";
+ dev_class->type = FP_DEVICE_TYPE_USB;
+ dev_class->scan_type = FP_SCAN_TYPE_PRESS;
+ dev_class->id_table = id_table;
+ dev_class->nr_enroll_stages = DEFAULT_ENROLL_SAMPLES;
+
+ dev_class->open = gx_fp_init;
+ dev_class->close = gx_fp_exit;
+ dev_class->probe = gx_fp_probe;
+ dev_class->enroll = gx_fp_enroll;
+ dev_class->delete = gx_fp_template_delete;
+ dev_class->list = gx_fp_template_list;
+ dev_class->cancel = gx_fp_cancel;
+ dev_class->verify = gx_fp_verify_identify;
+ dev_class->identify = gx_fp_verify_identify;
+}
diff --git a/libfprint/drivers/goodixmoc/goodix.h b/libfprint/drivers/goodixmoc/goodix.h
new file mode 100644
index 0000000..c38bf9b
--- /dev/null
+++ b/libfprint/drivers/goodixmoc/goodix.h
@@ -0,0 +1,58 @@
+/*
+ * Goodix Moc driver for libfprint
+ * Copyright (C) 2019 Shenzhen Goodix Technology Co., Ltd.
+ *
+ * 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"
+
+G_DECLARE_FINAL_TYPE (FpiDeviceGoodixMoc, fpi_device_goodixmoc, FPI, DEVICE_GOODIXMOC, FpDevice)
+
+typedef enum {
+ FP_CMD_SEND = 0,
+ FP_CMD_GET_ACK,
+ FP_CMD_GET_DATA,
+ FP_CMD_NUM_STATES,
+} FpCmdState;
+
+
+typedef enum {
+ FP_INIT_VERSION = 0,
+ FP_INIT_CONFIG,
+ FP_INIT_NUM_STATES,
+} FpInitState;
+
+
+typedef enum {
+ FP_ENROLL_ENUM = 0,
+ FP_ENROLL_IDENTIFY,
+ FP_ENROLL_CREATE,
+ FP_ENROLL_CAPTURE,
+ FP_ENROLL_UPDATE,
+ FP_ENROLL_WAIT_FINGER_UP,
+ FP_ENROLL_CHECK_DUPLICATE,
+ FP_ENROLL_COMMIT,
+ FP_ENROLL_NUM_STATES,
+} FpEnrollState;
+
+typedef enum {
+ FP_VERIFY_CAPTURE = 0,
+ FP_VERIFY_IDENTIFY,
+ FP_VERIFY_NUM_STATES,
+} FpVerifyState;
diff --git a/libfprint/drivers/goodixmoc/goodix_proto.c b/libfprint/drivers/goodixmoc/goodix_proto.c
new file mode 100644
index 0000000..3174170
--- /dev/null
+++ b/libfprint/drivers/goodixmoc/goodix_proto.c
@@ -0,0 +1,422 @@
+/*
+ * Goodix Moc driver for libfprint
+ * Copyright (C) 2019 Shenzhen Goodix Technology Co., Ltd.
+ *
+ * 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
+ */
+
+#include <glib.h>
+#include "goodix_proto.h"
+
+/*
+ * Crc functions
+ */
+
+#define WIDTH (8 * sizeof (uint32_t))
+#define FINAL_XOR_VALUE 0xFFFFFFFF
+#define REFLECT_DATA(X) ((uint8_t) reflect ((X), 8))
+#define REFLECT_REMAINDER(X) ((unsigned int) reflect ((X), WIDTH))
+
+
+uint8_t
+gx_proto_crc8_calc (uint8_t *lubp_date, uint32_t lui_len)
+{
+ const uint8_t *data = lubp_date;
+ unsigned int crc = 0;
+ int i, j;
+
+ for (j = lui_len; j; j--, data++)
+ {
+ crc ^= (*data << 8);
+ for (i = 8; i; i--)
+ {
+ if (crc & 0x8000)
+ crc ^= (0x1070 << 3);
+ crc <<= 1;
+ }
+ }
+
+ crc >>= 8;
+ crc = ~crc;
+ return (uint8_t) crc;
+}
+
+typedef struct
+{
+ uint32_t crc;
+} gf_crc32_context;
+
+static uint32_t s_crc_table[256] =
+{ 0x0, 0x4c11db7, 0x9823b6e, 0xd4326d9, 0x130476dc, 0x17c56b6b, 0x1a864db2, 0x1e475005,
+ 0x2608edb8, 0x22c9f00f, 0x2f8ad6d6, 0x2b4bcb61, 0x350c9b64, 0x31cd86d3, 0x3c8ea00a, 0x384fbdbd,
+ 0x4c11db70, 0x48d0c6c7, 0x4593e01e, 0x4152fda9, 0x5f15adac, 0x5bd4b01b, 0x569796c2, 0x52568b75,
+ 0x6a1936c8, 0x6ed82b7f, 0x639b0da6, 0x675a1011, 0x791d4014, 0x7ddc5da3, 0x709f7b7a, 0x745e66cd,
+ 0x9823b6e0, 0x9ce2ab57, 0x91a18d8e, 0x95609039, 0x8b27c03c, 0x8fe6dd8b, 0x82a5fb52, 0x8664e6e5,
+ 0xbe2b5b58, 0xbaea46ef, 0xb7a96036, 0xb3687d81, 0xad2f2d84, 0xa9ee3033, 0xa4ad16ea, 0xa06c0b5d,
+ 0xd4326d90, 0xd0f37027, 0xddb056fe, 0xd9714b49, 0xc7361b4c, 0xc3f706fb, 0xceb42022, 0xca753d95,
+ 0xf23a8028, 0xf6fb9d9f, 0xfbb8bb46, 0xff79a6f1, 0xe13ef6f4, 0xe5ffeb43, 0xe8bccd9a, 0xec7dd02d,
+ 0x34867077, 0x30476dc0, 0x3d044b19, 0x39c556ae, 0x278206ab, 0x23431b1c, 0x2e003dc5, 0x2ac12072,
+ 0x128e9dcf, 0x164f8078, 0x1b0ca6a1, 0x1fcdbb16, 0x18aeb13, 0x54bf6a4, 0x808d07d, 0xcc9cdca,
+ 0x7897ab07, 0x7c56b6b0, 0x71159069, 0x75d48dde, 0x6b93dddb, 0x6f52c06c, 0x6211e6b5, 0x66d0fb02,
+ 0x5e9f46bf, 0x5a5e5b08, 0x571d7dd1, 0x53dc6066, 0x4d9b3063, 0x495a2dd4, 0x44190b0d, 0x40d816ba,
+ 0xaca5c697, 0xa864db20, 0xa527fdf9, 0xa1e6e04e, 0xbfa1b04b, 0xbb60adfc, 0xb6238b25, 0xb2e29692,
+ 0x8aad2b2f, 0x8e6c3698, 0x832f1041, 0x87ee0df6, 0x99a95df3, 0x9d684044, 0x902b669d, 0x94ea7b2a,
+ 0xe0b41de7, 0xe4750050, 0xe9362689, 0xedf73b3e, 0xf3b06b3b, 0xf771768c, 0xfa325055, 0xfef34de2,
+ 0xc6bcf05f, 0xc27dede8, 0xcf3ecb31, 0xcbffd686, 0xd5b88683, 0xd1799b34, 0xdc3abded, 0xd8fba05a,
+ 0x690ce0ee, 0x6dcdfd59, 0x608edb80, 0x644fc637, 0x7a089632, 0x7ec98b85, 0x738aad5c, 0x774bb0eb,
+ 0x4f040d56, 0x4bc510e1, 0x46863638, 0x42472b8f, 0x5c007b8a, 0x58c1663d, 0x558240e4, 0x51435d53,
+ 0x251d3b9e, 0x21dc2629, 0x2c9f00f0, 0x285e1d47, 0x36194d42, 0x32d850f5, 0x3f9b762c, 0x3b5a6b9b,
+ 0x315d626, 0x7d4cb91, 0xa97ed48, 0xe56f0ff, 0x1011a0fa, 0x14d0bd4d, 0x19939b94, 0x1d528623,
+ 0xf12f560e, 0xf5ee4bb9, 0xf8ad6d60, 0xfc6c70d7, 0xe22b20d2, 0xe6ea3d65, 0xeba91bbc, 0xef68060b,
+ 0xd727bbb6, 0xd3e6a601, 0xdea580d8, 0xda649d6f, 0xc423cd6a, 0xc0e2d0dd, 0xcda1f604, 0xc960ebb3,
+ 0xbd3e8d7e, 0xb9ff90c9, 0xb4bcb610, 0xb07daba7, 0xae3afba2, 0xaafbe615, 0xa7b8c0cc, 0xa379dd7b,
+ 0x9b3660c6, 0x9ff77d71, 0x92b45ba8, 0x9675461f, 0x8832161a, 0x8cf30bad, 0x81b02d74, 0x857130c3,
+ 0x5d8a9099, 0x594b8d2e, 0x5408abf7, 0x50c9b640, 0x4e8ee645, 0x4a4ffbf2, 0x470cdd2b, 0x43cdc09c,
+ 0x7b827d21, 0x7f436096, 0x7200464f, 0x76c15bf8, 0x68860bfd, 0x6c47164a, 0x61043093, 0x65c52d24,
+ 0x119b4be9, 0x155a565e, 0x18197087, 0x1cd86d30, 0x29f3d35, 0x65e2082, 0xb1d065b, 0xfdc1bec,
+ 0x3793a651, 0x3352bbe6, 0x3e119d3f, 0x3ad08088, 0x2497d08d, 0x2056cd3a, 0x2d15ebe3, 0x29d4f654,
+ 0xc5a92679, 0xc1683bce, 0xcc2b1d17, 0xc8ea00a0, 0xd6ad50a5, 0xd26c4d12, 0xdf2f6bcb, 0xdbee767c,
+ 0xe3a1cbc1, 0xe760d676, 0xea23f0af, 0xeee2ed18, 0xf0a5bd1d, 0xf464a0aa, 0xf9278673, 0xfde69bc4,
+ 0x89b8fd09, 0x8d79e0be, 0x803ac667, 0x84fbdbd0, 0x9abc8bd5, 0x9e7d9662, 0x933eb0bb, 0x97ffad0c,
+ 0xafb010b1, 0xab710d06, 0xa6322bdf, 0xa2f33668, 0xbcb4666d, 0xb8757bda, 0xb5365d03, 0xb1f740b4};
+
+static uint32_t
+reflect (uint32_t data, uint8_t n_bits)
+{
+ unsigned long reflection = 0x00000000;
+ uint8_t bit;
+
+ /*
+ * Reflect the data about the center bit.
+ */
+ for (bit = 0; bit < n_bits; ++bit)
+ {
+ /*
+ * If the LSB bit is set, set the reflection of it.
+ */
+ if (data & 0x01)
+ reflection |= (1 << ((n_bits - 1) - bit));
+
+ data = (data >> 1);
+ }
+
+ return reflection;
+
+}
+
+static void
+crc32_init (gf_crc32_context *ctx)
+{
+ ctx->crc = 0xFFFFFFFF;
+}
+
+static void
+crc32_update (gf_crc32_context *ctx, const uint8_t *message, uint32_t n_bytes)
+{
+ uint8_t data;
+ uint32_t byte;
+
+ /*
+ * Divide the message by the polynomial, a byte at a time.
+ */
+ for (byte = 0; byte < n_bytes; ++byte)
+ {
+ data = REFLECT_DATA (message[byte]) ^ (ctx->crc >> (WIDTH - 8));
+ ctx->crc = s_crc_table[data] ^ (ctx->crc << 8);
+ }
+}
+
+static void
+crc32_final (gf_crc32_context *ctx, uint8_t *md)
+{
+ ctx->crc = (REFLECT_REMAINDER (ctx->crc) ^ FINAL_XOR_VALUE);
+ memcpy (md, &ctx->crc, 4);
+}
+
+uint8_t
+gx_proto_crc32_calc (uint8_t *pchMsg, uint32_t wDataLen, uint8_t *pchMsgDst)
+{
+ gf_crc32_context context = { 0 };
+
+ if (!pchMsg)
+ return 0;
+
+ crc32_init (&context);
+
+ crc32_update (&context, pchMsg, wDataLen);
+
+ crc32_final (&context, pchMsgDst);
+
+
+ return 1;
+}
+/*
+ * protocol
+ *
+ */
+
+static uint8_t dump_seq = 0;
+
+static void
+init_pack_header (
+ ppack_header pheader,
+ uint16_t len,
+ uint16_t cmd,
+ uint8_t packagenum
+ )
+{
+ g_assert (pheader);
+
+ memset (pheader, 0, sizeof (*pheader));
+ pheader->cmd0 = HIBYTE (cmd);
+ pheader->cmd1 = LOBYTE (cmd);
+ pheader->packagenum = packagenum;
+ pheader->reserved = dump_seq++;
+ pheader->len = len + PACKAGE_CRC_SIZE;
+ pheader->crc8 = gx_proto_crc8_calc ((uint8_t *) pheader, 6);
+ pheader->rev_crc8 = ~pheader->crc8;
+}
+
+
+int
+gx_proto_build_package (uint8_t *ppackage,
+ uint32_t *package_len,
+ uint16_t cmd,
+ const uint8_t *payload,
+ uint32_t payload_size)
+{
+ pack_header header;
+
+ if (!ppackage || !package_len)
+ return -1;
+
+ if(*package_len < (payload_size + PACKAGE_HEADER_SIZE + PACKAGE_CRC_SIZE))
+ return -1;
+
+ init_pack_header (&header, payload_size, cmd, 0);
+
+ memcpy (ppackage, &header, PACKAGE_HEADER_SIZE);
+ memcpy (ppackage + PACKAGE_HEADER_SIZE, payload, payload_size);
+
+ gx_proto_crc32_calc (ppackage, PACKAGE_HEADER_SIZE + payload_size, ppackage + PACKAGE_HEADER_SIZE + payload_size);
+
+ return 0;
+}
+
+
+int
+gx_proto_parse_header (
+ uint8_t *buffer,
+ uint32_t buffer_len,
+ pack_header *pheader)
+{
+ if (!buffer || !pheader)
+ return -1;
+ if (buffer_len < PACKAGE_HEADER_SIZE)
+ return -1;
+
+ memcpy (pheader, buffer, sizeof (pack_header));
+
+ pheader->len = GUINT16_FROM_LE (*(buffer + 4));
+ pheader->len -= PACKAGE_CRC_SIZE;
+
+ return 0;
+}
+
+static int
+gx_proto_parse_fingerid (
+ uint8_t * fid_buffer,
+ uint16_t fid_buffer_size,
+ ptemplate_format_t template
+ )
+{
+ uint8_t * buffer = NULL;
+ uint16_t Offset = 0;
+
+ if (!template || !fid_buffer)
+ return -1;
+
+ if (fid_buffer_size < 70)
+ return -1;
+
+ buffer = fid_buffer;
+ Offset = 0;
+
+ if (buffer[Offset++] != 67)
+ return -1;
+
+ template->type = buffer[Offset++];
+ template->finger_index = buffer[Offset++];
+ Offset++;
+
+ memcpy (template->accountid, &buffer[Offset], 32);
+ Offset += 32;
+
+ memcpy (template->tid, &buffer[Offset], 32);
+ Offset += 32; // Offset == 68
+
+ template->payload.size = buffer[Offset++];
+ memset (template->payload.data, 0, 56);
+ memcpy (template->payload.data, &buffer[Offset], template->payload.size);
+
+ return 0;
+}
+
+int
+gx_proto_parse_body (uint16_t cmd, uint8_t *buffer, uint32_t buffer_len, pgxfp_cmd_response_t presp)
+{
+ uint32_t offset = 0;
+ uint8_t *fingerlist = NULL;
+
+ if (!buffer || !presp)
+ return -1;
+ if (buffer_len < 1)
+ return -1;
+ presp->result = buffer[0];
+ switch (HIBYTE (cmd))
+ {
+ case RESPONSE_PACKAGE_CMD:
+ {
+ presp->parse_msg.ack_cmd = buffer[1];
+ }
+ break;
+
+ case MOC_CMD0_UPDATE_CONFIG:
+ case MOC_CMD0_COMMITENROLLMENT:
+ case MOC_CMD0_DELETETEMPLATE:
+ break;
+
+ case MOC_CMD0_GET_VERSION:
+ memcpy (&presp->version_info, buffer + 1, sizeof (gxfp_version_info_t));
+ break;
+
+ case MOC_CMD0_CAPTURE_DATA:
+ if (LOBYTE (cmd) == MOC_CMD1_DEFAULT)
+ {
+ presp->capture_data_resp.img_quality = buffer[1];
+ presp->capture_data_resp.img_coverage = buffer[2];
+ }
+ break;
+
+ case MOC_CMD0_ENROLL_INIT:
+ if (presp->result == GX_SUCCESS)
+ memcpy (&presp->enroll_init.tid, &buffer[1], TEMPLATE_ID_SIZE);
+ break;
+
+ case MOC_CMD0_ENROLL:
+ presp->enroll_update.rollback = (buffer[0] < 0x80) ? false : true;
+ presp->enroll_update.img_overlay = buffer[1];
+ presp->enroll_update.img_preoverlay = buffer[2];
+ break;
+
+ case MOC_CMD0_CHECK4DUPLICATE:
+ presp->check_duplicate_resp.duplicate = (presp->result == 0) ? false : true;
+ if (presp->check_duplicate_resp.duplicate)
+ {
+ uint16_t tid_size = GUINT16_FROM_LE (*(buffer + 1));
+ memcpy (&presp->check_duplicate_resp.template, buffer + 3, tid_size);
+ }
+ break;
+
+ case MOC_CMD0_GETFINGERLIST:
+ if (presp->result != GX_SUCCESS)
+ break;
+ presp->finger_list_resp.finger_num = buffer[1];
+ if (presp->finger_list_resp.finger_num > FP_MAX_FINGERNUM)
+ {
+ presp->finger_list_resp.finger_num = 0;
+ presp->result = GX_ERROR_NO_AVAILABLE_SPACE;
+ break;
+ }
+ fingerlist = buffer + 2;
+ for(uint8_t num = 0; num < presp->finger_list_resp.finger_num; num++)
+ {
+ uint16_t fingerid_length = GUINT16_FROM_LE (*(fingerlist + offset));
+ offset += 2;
+ if (gx_proto_parse_fingerid (fingerlist + offset,
+ fingerid_length,
+ &presp->finger_list_resp.finger_list[num]) != 0)
+ {
+ g_error ("parse fingerlist error");
+ presp->finger_list_resp.finger_num = 0;
+ presp->result = GX_FAILED;
+ break;
+ }
+ offset += fingerid_length;
+ }
+ break;
+
+ case MOC_CMD0_IDENTIFY:
+ {
+ uint32_t score = 0;
+ uint8_t study = 0;
+ uint16_t fingerid_size = 0;
+ presp->verify.match = (buffer[0] == 0) ? true : false;
+ if (presp->verify.match)
+ {
+ offset += 1;
+ presp->verify.rejectdetail = GUINT16_FROM_LE (*(buffer + offset));
+ offset += 2;
+ score = GUINT32_FROM_LE (*(buffer + offset));
+ offset += 4;
+ study = GUINT16_FROM_LE (*(buffer + offset));
+ offset += 1;
+ fingerid_size = GUINT16_FROM_LE (*(buffer + offset));
+ offset += 2;
+ if (gx_proto_parse_fingerid (buffer + offset, fingerid_size, &presp->verify.template) != 0)
+ {
+ presp->result = GX_FAILED;
+ break;
+ }
+ g_debug ("match, score: %d, study: %d", score, study);
+ }
+ }
+ break;
+
+ case MOC_CMD0_FINGER_MODE:
+ presp->finger_status.status = buffer[0];
+ break;
+
+ default:
+ break;
+ }
+
+ return 0;
+}
+
+static uint8_t sensor_config[26] = {
+ 0x00, 0x00, 0x64, 0x50, 0x0f, 0x41, 0x08, 0x0a, 0x18, 0x00, 0x00, 0x23, 0x00,
+ 0x00, 0x01, 0x01, 0x00, 0x01, 0x01, 0x01, 0x01, 0x00, 0x01, 0x01, 0x05, 0x05
+};
+
+int
+gx_proto_init_sensor_config (pgxfp_sensor_cfg_t pconfig)
+{
+ uint32_t crc32_calc = 0;
+
+ if (!pconfig)
+ return -1;
+ memset (pconfig, 0, sizeof (*pconfig));
+
+ //NOTICE: Do not change any value!
+ memcpy (&pconfig->config, sensor_config, 26);
+ pconfig->reserved[0] = 1;
+
+ gx_proto_crc32_calc ((uint8_t *) pconfig, sizeof (*pconfig) - PACKAGE_CRC_SIZE, (uint8_t *) &crc32_calc);
+
+ memcpy (pconfig->crc_value, &crc32_calc, PACKAGE_CRC_SIZE);
+
+ return 0;
+} \ No newline at end of file
diff --git a/libfprint/drivers/goodixmoc/goodix_proto.h b/libfprint/drivers/goodixmoc/goodix_proto.h
new file mode 100644
index 0000000..1611a60
--- /dev/null
+++ b/libfprint/drivers/goodixmoc/goodix_proto.h
@@ -0,0 +1,232 @@
+/*
+ * Goodix Moc driver for libfprint
+ * Copyright (C) 2019 Shenzhen Goodix Technology Co., Ltd.
+ *
+ * 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 <stdint.h>
+#include <stdbool.h>
+#include <string.h>
+
+#define PACKAGE_CRC_SIZE (4)
+#define PACKAGE_HEADER_SIZE (8)
+
+#define FP_MAX_FINGERNUM (10)
+
+#define TEMPLATE_ID_SIZE (32)
+
+#define GX_VERSION_LEN (8)
+
+/* Type covert */
+#define MAKE_CMD_EX(cmd0, cmd1) ((uint16_t) (((cmd0) << 8) | (cmd1)))
+#define LOBYTE(value) ((uint8_t) (value))
+#define HIBYTE(value) ((uint8_t) (((uint16_t) (value) >> 8) & 0xFF))
+
+
+/* Error code */
+#define GX_SUCCESS 0x00
+#define GX_FAILED 0x80
+#define GX_ERROR_FINGER_ID_NOEXIST 0x9C
+#define GX_ERROR_TEMPLATE_INCOMPLETE 0xB8
+#define GX_ERROR_WAIT_FINGER_UP_TIMEOUT 0xC7
+#define GX_ERROR_NO_AVAILABLE_SPACE 0x8F
+
+/* Command Type Define */
+#define RESPONSE_PACKAGE_CMD 0xAA
+
+#define MOC_CMD0_ENROLL 0xA0
+#define MOC_CMD0_ENROLL_INIT 0xA1
+#define MOC_CMD0_CAPTURE_DATA 0xA2
+#define MOC_CMD0_CHECK4DUPLICATE 0xA3
+#define MOC_CMD0_COMMITENROLLMENT 0xA4
+
+#define MOC_CMD0_IDENTIFY 0xA5
+#define MOC_CMD0_GETFINGERLIST 0xA6
+#define MOC_CMD0_DELETETEMPLATE 0xA7
+
+#define MOC_CMD1_DEFAULT 0x00
+#define MOC_CMD1_UNTIL_DOWN 0x00
+#define MOC_CMD1_WHEN_DOWN 0x01
+
+#define MOC_CMD1_DELETE_TEMPLATE 0x04
+#define MOC_CMD1_DELETE_ALL 0x01
+
+#define MOC_CMD0_GET_VERSION 0xD0
+
+#define MOC_CMD0_UPDATE_CONFIG 0xC0
+#define MOC_CMD1_NWRITE_CFG_TO_FLASH 0x00
+#define MOC_CMD1_WRITE_CFG_TO_FLASH 0x01
+
+#define MOC_CMD0_FINGER_MODE 0xB0
+#define MOC_CMD1_GET_FINGER_MODE 0x00
+#define MOC_CMD1_SET_FINGER_DOWN 0x01
+#define MOC_CMD1_SET_FINGER_UP 0x02
+/* */
+
+typedef struct _gxfp_version_info
+{
+ uint8_t format[2];
+ uint8_t fwtype[GX_VERSION_LEN];
+ uint8_t fwversion[GX_VERSION_LEN];
+ uint8_t customer[GX_VERSION_LEN];
+ uint8_t mcu[GX_VERSION_LEN];
+ uint8_t sensor[GX_VERSION_LEN];
+ uint8_t algversion[GX_VERSION_LEN];
+ uint8_t interface[GX_VERSION_LEN];
+ uint8_t protocol[GX_VERSION_LEN];
+ uint8_t flashVersion[GX_VERSION_LEN];
+ uint8_t reserved[62];
+} gxfp_version_info_t, *pgxfp_version_info_t;
+
+
+typedef struct _gxfp_parse_msg
+{
+ uint8_t ack_cmd;
+ bool has_no_config;
+} gxfp_parse_msg_t, *pgxfp_parse_msg_t;
+
+
+typedef struct _gxfp_enroll_init
+{
+ uint8_t tid[TEMPLATE_ID_SIZE];
+} gxfp_enroll_init_t, *pgxfp_enroll_init_t;
+
+#pragma pack(push, 1)
+typedef struct _template_format
+{
+ uint8_t type;
+ uint8_t finger_index;
+ uint8_t accountid[32];
+ uint8_t tid[32];
+ struct
+ {
+ uint32_t size;
+ uint8_t data[56];
+ } payload;
+ uint8_t reserve[2];
+} template_format_t, *ptemplate_format_t;
+
+#pragma pack(pop)
+
+
+typedef struct _gxfp_verify
+{
+ bool match;
+ uint32_t rejectdetail;
+ template_format_t template;
+} gxfp_verify_t, *pgxfp_verify_t;
+
+
+typedef struct _gxfp_capturedata
+{
+ uint8_t img_quality;
+ uint8_t img_coverage;
+} gxfp_capturedata_t, *pgxfp_capturedata_t;
+
+typedef struct _gxfp_check_duplicate
+{
+ bool duplicate;
+ template_format_t template;
+} gxfp_check_duplicate_t, *pgxfp_check_duplicate_t;
+
+
+typedef struct _gxfp_enroll_update
+{
+ bool rollback;
+ uint8_t img_overlay;
+ uint8_t img_preoverlay;
+} gxfp_enroll_update_t, *Pgxfp_enroll_update_t;
+
+typedef struct _gxfp_enum_fingerlist
+{
+ uint8_t finger_num;
+ template_format_t finger_list[FP_MAX_FINGERNUM];
+} gxfp_enum_fingerlist_t, *pgxfp_enum_fingerlist_t;
+
+typedef struct _gxfp_enroll_commit
+{
+ uint8_t result;
+} gxfp_enroll_commit_t, *pgxfp_enroll_commit_t;
+
+typedef struct _fp_finger_status
+{
+ uint8_t status;
+} fp_finger_status_t, *pfp_finger_status_t;
+
+
+typedef struct _fp_cmd_response
+{
+ uint8_t result;
+ union
+ {
+ gxfp_parse_msg_t parse_msg;
+ gxfp_verify_t verify;
+ gxfp_enroll_init_t enroll_init;
+ gxfp_capturedata_t capture_data_resp;
+ gxfp_check_duplicate_t check_duplicate_resp;
+ gxfp_enroll_commit_t enroll_commit;
+ gxfp_enroll_update_t enroll_update;
+ gxfp_enum_fingerlist_t finger_list_resp;
+ gxfp_version_info_t version_info;
+ fp_finger_status_t finger_status;
+ };
+} gxfp_cmd_response_t, *pgxfp_cmd_response_t;
+
+
+typedef struct _pack_header
+{
+ uint8_t cmd0;
+ uint8_t cmd1;
+ uint8_t packagenum;
+ uint8_t reserved;
+ uint16_t len;
+ uint8_t crc8;
+ uint8_t rev_crc8;
+} pack_header, *ppack_header;
+
+
+typedef struct _gxfp_sensor_cfg
+{
+ uint8_t config[26];
+ uint8_t reserved[98];
+ uint8_t crc_value[4];
+} gxfp_sensor_cfg_t, *pgxfp_sensor_cfg_t;
+/* */
+
+int gx_proto_build_package (uint8_t *ppackage,
+ uint32_t *package_len,
+ uint16_t cmd,
+ const uint8_t *payload,
+ uint32_t payload_size);
+
+int gx_proto_parse_header (uint8_t *buffer,
+ uint32_t buffer_len,
+ pack_header *pheader);
+
+int gx_proto_parse_body (uint16_t cmd,
+ uint8_t *buffer,
+ uint32_t buffer_len,
+ pgxfp_cmd_response_t presponse);
+
+int gx_proto_init_sensor_config (pgxfp_sensor_cfg_t pconfig);
+
+uint8_t gx_proto_crc8_calc (uint8_t *lubp_date,
+ uint32_t lui_len);
+
+uint8_t gx_proto_crc32_calc (uint8_t *pchMsg,
+ uint32_t wDataLen,
+ uint8_t *pchMsgDst);
diff --git a/libfprint/drivers/synaptics/synaptics.c b/libfprint/drivers/synaptics/synaptics.c
index 25ae162..9bbae99 100644
--- a/libfprint/drivers/synaptics/synaptics.c
+++ b/libfprint/drivers/synaptics/synaptics.c
@@ -29,7 +29,8 @@ G_DEFINE_TYPE (FpiDeviceSynaptics, fpi_device_synaptics, FP_TYPE_DEVICE)
static const FpIdEntry id_table[] = {
{ .vid = SYNAPTICS_VENDOR_ID, .pid = 0xBD, },
-
+ { .vid = SYNAPTICS_VENDOR_ID, .pid = 0xE9, },
+ { .vid = SYNAPTICS_VENDOR_ID, .pid = 0xDF, },
{ .vid = 0, .pid = 0, .driver_data = 0 }, /* terminating entry */
};
@@ -897,6 +898,7 @@ dev_probe (FpDevice *device)
const guint8 *data;
gboolean read_ok = TRUE;
g_autofree gchar *serial = NULL;
+ gboolean retry = TRUE;
G_DEBUG_HERE ();
@@ -914,35 +916,43 @@ dev_probe (FpDevice *device)
if (!g_usb_device_claim_interface (usb_dev, 0, 0, &error))
goto err_close;
- /* TODO: Do not do this synchronous. */
- transfer = fpi_usb_transfer_new (device);
- fpi_usb_transfer_fill_bulk (transfer, USB_EP_REQUEST, SENSOR_FW_CMD_HEADER_LEN);
- transfer->short_is_error = TRUE;
- transfer->buffer[0] = SENSOR_CMD_GET_VERSION;
- if (!fpi_usb_transfer_submit_sync (transfer, 1000, &error))
- goto err_close;
+ while(1)
+ {
+ /* TODO: Do not do this synchronous. */
+ transfer = fpi_usb_transfer_new (device);
+ fpi_usb_transfer_fill_bulk (transfer, USB_EP_REQUEST, SENSOR_FW_CMD_HEADER_LEN);
+ transfer->short_is_error = TRUE;
+ transfer->buffer[0] = SENSOR_CMD_GET_VERSION;
+ if (!fpi_usb_transfer_submit_sync (transfer, 1000, &error))
+ goto err_close;
- g_clear_pointer (&transfer, fpi_usb_transfer_unref);
- transfer = fpi_usb_transfer_new (device);
- fpi_usb_transfer_fill_bulk (transfer, USB_EP_REPLY, 40);
- if (!fpi_usb_transfer_submit_sync (transfer, 1000, &error))
- goto err_close;
+ g_clear_pointer (&transfer, fpi_usb_transfer_unref);
+ transfer = fpi_usb_transfer_new (device);
+ fpi_usb_transfer_fill_bulk (transfer, USB_EP_REPLY, 40);
+ if (!fpi_usb_transfer_submit_sync (transfer, 1000, &error))
+ goto err_close;
- fpi_byte_reader_init (&reader, transfer->buffer, transfer->actual_length);
+ fpi_byte_reader_init (&reader, transfer->buffer, transfer->actual_length);
- if (!fpi_byte_reader_get_uint16_le (&reader, &status))
- {
- g_warning ("Transfer in response to version query was too short");
- error = fpi_device_error_new (FP_DEVICE_ERROR_PROTO);
- goto err_close;
- }
- if (status != 0)
- {
- g_warning ("Device responded with error: %d", status);
- error = fpi_device_error_new (FP_DEVICE_ERROR_PROTO);
- goto err_close;
+ if (!fpi_byte_reader_get_uint16_le (&reader, &status))
+ {
+ g_warning ("Transfer in response to version query was too short");
+ error = fpi_device_error_new (FP_DEVICE_ERROR_PROTO);
+ goto err_close;
+ }
+ if (status != 0)
+ {
+ g_warning ("Device responded with error: %d retry: %d", status, retry);
+ if(retry)
+ {
+ retry = FALSE;
+ continue;
+ }
+ error = fpi_device_error_new (FP_DEVICE_ERROR_PROTO);
+ goto err_close;
+ }
+ break;
}
-
read_ok &= fpi_byte_reader_get_uint32_le (&reader, &self->mis_version.build_time);
read_ok &= fpi_byte_reader_get_uint32_le (&reader, &self->mis_version.build_num);
read_ok &= fpi_byte_reader_get_uint8 (&reader, &self->mis_version.version_major);
diff --git a/libfprint/fp-context.c b/libfprint/fp-context.c
index 12cf4e6..6dadc62 100644
--- a/libfprint/fp-context.c
+++ b/libfprint/fp-context.c
@@ -215,7 +215,8 @@ fp_context_finalize (GObject *object)
g_clear_object (&priv->cancellable);
g_clear_pointer (&priv->drivers, g_array_unref);
- g_object_run_dispose (G_OBJECT (priv->usb_ctx));
+ if (priv->usb_ctx)
+ g_object_run_dispose (G_OBJECT (priv->usb_ctx));
g_clear_object (&priv->usb_ctx);
fpi_tod_shared_drivers_unregister ();
@@ -301,7 +302,7 @@ fp_context_init (FpContext *self)
priv->usb_ctx = g_usb_context_new (&error);
if (!priv->usb_ctx)
{
- fp_warn ("Could not initialise USB Subsystem: %s", error->message);
+ g_message ("Could not initialise USB Subsystem: %s", error->message);
}
else
{
@@ -354,7 +355,8 @@ fp_context_enumerate (FpContext *context)
priv->enumerated = TRUE;
/* USB devices are handled from callbacks */
- g_usb_context_enumerate (priv->usb_ctx);
+ if (priv->usb_ctx)
+ g_usb_context_enumerate (priv->usb_ctx);
/* Handle Virtual devices based on environment variables */
for (i = 0; i < priv->drivers->len; i++)
diff --git a/libfprint/fp-device.h b/libfprint/fp-device.h
index 5c435fd..0be64ec 100644
--- a/libfprint/fp-device.h
+++ b/libfprint/fp-device.h
@@ -90,6 +90,7 @@ typedef enum {
* @FP_DEVICE_ERROR_DATA_INVALID: The passed data is invalid
* @FP_DEVICE_ERROR_DATA_NOT_FOUND: Requested print was not found on device
* @FP_DEVICE_ERROR_DATA_FULL: No space on device available for operation
+ * @FP_DEVICE_ERROR_DATA_DUPLICATE: Enrolling template duplicates storaged templates
*
* 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.
@@ -104,6 +105,7 @@ typedef enum {
FP_DEVICE_ERROR_DATA_INVALID,
FP_DEVICE_ERROR_DATA_NOT_FOUND,
FP_DEVICE_ERROR_DATA_FULL,
+ FP_DEVICE_ERROR_DATA_DUPLICATE,
} FpDeviceError;
GQuark fp_device_retry_quark (void);
diff --git a/libfprint/fp-image-device.h b/libfprint/fp-image-device.h
index 1eda8fb..e23148b 100644
--- a/libfprint/fp-image-device.h
+++ b/libfprint/fp-image-device.h
@@ -19,7 +19,7 @@
#pragma once
-#include <fp-device.h>
+#include "fp-device.h"
G_BEGIN_DECLS
diff --git a/libfprint/fpi-device.c b/libfprint/fpi-device.c
index 7c1c851..e62ab19 100644
--- a/libfprint/fpi-device.c
+++ b/libfprint/fpi-device.c
@@ -135,6 +135,10 @@ fpi_device_error_new (FpDeviceError error)
msg = "Print was not found on the devices storage.";
break;
+ case FP_DEVICE_ERROR_DATA_DUPLICATE:
+ msg = "This finger has already enrolled, please try a different finger";
+ break;
+
default:
g_warning ("Unsupported error, returning general error instead!");
error = FP_DEVICE_ERROR_GENERAL;
@@ -391,7 +395,7 @@ fpi_device_action_is_cancelled (FpDevice *device)
cancellable = g_task_get_cancellable (priv->current_task);
- return cancellable ? g_cancellable_is_cancelled (cancellable) : FALSE;
+ return g_cancellable_is_cancelled (cancellable);
}
/**
@@ -593,7 +597,11 @@ fpi_device_action_error (FpDevice *device,
if (error != NULL)
{
- g_debug ("Device reported generic error during action; action was: %i", priv->current_action);
+ g_autofree char *action_str = NULL;
+
+ action_str = g_enum_to_string (FPI_TYPE_DEVICE_ACTION, priv->current_action);
+ g_debug ("Device reported generic error (%s) during action; action was: %s",
+ error->message, action_str);
}
else
{
@@ -682,10 +690,13 @@ fp_device_task_return_in_idle_cb (gpointer user_data)
{
FpDeviceTaskReturnData *data = user_data;
FpDevicePrivate *priv = fp_device_get_instance_private (data->device);
+ g_autofree char *action_str = NULL;
g_autoptr(GTask) task = NULL;
- g_debug ("Completing action %d in idle!", priv->current_action);
+
+ action_str = g_enum_to_string (FPI_TYPE_DEVICE_ACTION, priv->current_action);
+ g_debug ("Completing action %s in idle!", action_str);
task = g_steal_pointer (&priv->current_task);
priv->current_action = FPI_DEVICE_ACTION_NONE;
diff --git a/libfprint/fpi-image-device.c b/libfprint/fpi-image-device.c
index 11399a1..8fc1b92 100644
--- a/libfprint/fpi-image-device.c
+++ b/libfprint/fpi-image-device.c
@@ -94,6 +94,8 @@ static void
fp_image_device_change_state (FpImageDevice *self, FpiImageDeviceState state)
{
FpImageDevicePrivate *priv = fp_image_device_get_instance_private (self);
+ g_autofree char *prev_state_str = NULL;
+ g_autofree char *state_str = NULL;
/* Cannot change to inactive using this function. */
g_assert (state != FPI_IMAGE_DEVICE_STATE_INACTIVE);
@@ -102,7 +104,10 @@ fp_image_device_change_state (FpImageDevice *self, FpiImageDeviceState state)
* next operation. */
g_clear_handle_id (&priv->pending_activation_timeout_id, g_source_remove);
- fp_dbg ("Image device internal state change from %d to %d\n", priv->state, state);
+ prev_state_str = g_enum_to_string (FPI_TYPE_IMAGE_DEVICE_STATE, priv->state);
+ state_str = g_enum_to_string (FPI_TYPE_IMAGE_DEVICE_STATE, state);
+ fp_dbg ("Image device internal state change from %s to %s\n",
+ prev_state_str, state_str);
priv->state = state;
g_object_notify (G_OBJECT (self), "fpi-image-device-state");
@@ -121,6 +126,7 @@ fp_image_device_enroll_maybe_await_finger_on (FpImageDevice *self)
}
else
{
+ fp_dbg ("Awaiting finger on");
priv->enroll_await_on_pending = TRUE;
}
}
diff --git a/libfprint/fpi-image-device.h b/libfprint/fpi-image-device.h
index 864cb90..c684879 100644
--- a/libfprint/fpi-image-device.h
+++ b/libfprint/fpi-image-device.h
@@ -72,9 +72,8 @@ typedef enum {
* fpi_device_action_error() function but doing so is not recommended in most
* usecases.
*
- * 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).
+ * Image drivers must expect a @deactivate call to happen at any point during
+ * image capture.
*
* This API is solely intended for drivers. It is purely internal and neither
* API nor ABI stable.
diff --git a/libfprint/fpi-print.c b/libfprint/fpi-print.c
index 617e7b4..16877b8 100644
--- a/libfprint/fpi-print.c
+++ b/libfprint/fpi-print.c
@@ -240,7 +240,7 @@ fpi_print_bz3_match (FpPrint *template, FpPrint *print, gint bz3_threshold, GErr
gint score;
gstruct = g_ptr_array_index (template->prints, i);
score = bozorth_to_gallery (probe_len, pstruct, gstruct);
- fp_dbg ("score %d", score);
+ fp_dbg ("score %d/%d", score, bz3_threshold);
if (score >= bz3_threshold)
return FPI_MATCH_SUCCESS;
diff --git a/libfprint/fpi-usb-transfer.c b/libfprint/fpi-usb-transfer.c
index 74b5b1c..37033c8 100644
--- a/libfprint/fpi-usb-transfer.c
+++ b/libfprint/fpi-usb-transfer.c
@@ -105,6 +105,7 @@ fpi_usb_transfer_new (FpDevice * device)
self = g_slice_new0 (FpiUsbTransfer);
self->ref_count = 1;
+ self->type = FP_TRANSFER_NONE;
self->device = device;
diff --git a/libfprint/fpi-usb-transfer.h b/libfprint/fpi-usb-transfer.h
index c905bff..2e6c03c 100644
--- a/libfprint/fpi-usb-transfer.h
+++ b/libfprint/fpi-usb-transfer.h
@@ -47,10 +47,10 @@ typedef void (*FpiUsbTransferCallback)(FpiUsbTransfer *transfer,
* Type of the transfer.
*/
typedef enum {
- FP_TRANSFER_NONE = 0,
- FP_TRANSFER_BULK,
- FP_TRANSFER_CONTROL,
- FP_TRANSFER_INTERRUPT,
+ FP_TRANSFER_NONE = -1,
+ FP_TRANSFER_CONTROL = 0,
+ FP_TRANSFER_BULK = 2,
+ FP_TRANSFER_INTERRUPT = 3,
} FpiTransferType;
/**
diff --git a/libfprint/meson.build b/libfprint/meson.build
index 78448ce..317e4c2 100644
--- a/libfprint/meson.build
+++ b/libfprint/meson.build
@@ -161,6 +161,12 @@ foreach driver: drivers
'drivers/synaptics/bmkt_message.c',
]
endif
+ if driver == 'goodixmoc'
+ drivers_sources += [
+ 'drivers/goodixmoc/goodix.c',
+ 'drivers/goodixmoc/goodix_proto.c',
+ ]
+ endif
endforeach
if aeslib
diff --git a/meson.build b/meson.build
index bab3f52..5c26395 100644
--- a/meson.build
+++ b/meson.build
@@ -1,5 +1,5 @@
project('libfprint', [ 'c', 'cpp' ],
- version: '1.90.2+tod1',
+ version: '1.90.3+tod1',
license: 'LGPLv2.1+',
default_options: [
'buildtype=debugoptimized',
@@ -110,6 +110,7 @@ default_drivers = [
'upektc',
'upeksonly',
'upekts',
+ 'goodixmoc',
]
all_drivers = default_drivers + virtual_drivers
diff --git a/tests/goodixmoc/custom.ioctl b/tests/goodixmoc/custom.ioctl
new file mode 100644
index 0000000..21c8531
--- /dev/null
+++ b/tests/goodixmoc/custom.ioctl
@@ -0,0 +1,307 @@
+@DEV /dev/bus/usb/001/003
+USBDEVFS_GET_CAPABILITIES 0 FD010000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 13 13 0 D00000000500BA4500611A297F
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00000008009D6200D00001B5A57582000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 192 0 D0000001850067980001FE415050000000000030313030303232384C454E4F564F0000474D3138384230004746333230360000312E30322E30310055534200000000005642530000000000303030303030303300000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000051239DE6303030303033000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 140 140 0 C001000184008E71000064500F410C0A1800002300000101000101010100010105050100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000B737316F3EB36C6A
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00000208004BB400C00101C96A6C6B000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 C00100030500FB040093B3ED01000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 13 13 0 A6000002050049B600C27E4B39
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA000004080036C900A60001F1AFC9FB000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A60000050600609F000094D6C40E0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A20000030700AC53000F411A349263
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA0000060800E01F00A20001605AE410000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A2000007070007F8005564FA6B157100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 44 44 0 A50000042400F40B000000000000000000000000000000000000000000000000000000000000000095D4A28A
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA0000080800CC3300A500013F9036A9000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A50000090500DE2103D6515435000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 13 13 0 A10000050500807F007BE269C4
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00000A08001AE500A10001AE651B42000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A100000B250002FD00B786B17D6A044D24C1651C2B1A76F6396D790639F58CA6D62DDDB8E179A9BD4A6C5C6C9200000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A200000607006C93000F41A33C2AB4
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00000C0800679800A200011C98B985000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A200000D0700807F0059643EDA283F00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A0000007070055AA016450891085EC
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00000E0800B14E00A000013F11196A000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A000000F070004FB000000E0109A2200000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 13 13 0 B00200080500629D004A354747
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00001008003FC000B002017532670A000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 128 0 B00200114500A15E008E0091009E006F0083007C006D00690079008800000000000000000000000000AF00A500AD00A000B100A300AA00A300A3009D00000000000000000000000000988D37C39E006F0083007C006D00690079008800000000000000000000000000AF00A500AD00A000B100A300AA00A300A3009D00000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A200000907002BD4000F41D2CA1A81
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA0000120800E91600A20001A47AD7CB000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A200001307000EF1004B640F46BD2D00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A000000A0700C43B016450398D1A9E
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA0000140800946B00A0000121EDB70D000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A0000015070021DE0000007BC22C5100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 13 13 0 B002000B0500DF200052FA3E5D
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA000016080042BD00B002019E7183CF000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 128 0 B00200174500DC23008200A200AA0093009F009800A90087009A009100000000000000000000000000AF00A500AD00A000B100A300AA00A300A3009C00000000000000000000000000F1B14CECAA0093009F009800A90087009A009100000000000000000000000000AF00A500AD00A000B100A300AA00A300A3009C00000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A200000C0700EB14000F416BC2A256
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00001808006E9100A20001D8B88A5E000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A200001907008976005664DBD4593000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A000000D0700D22D01645041EE0D0E
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00001A0800B84700A00001FB312AB1000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A000001B07000DF2000000CA0273AA00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 13 13 0 B002000E05001FE000EF2AA657
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00001C0800C53A00B00201E2B3DE5A000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 128 0 B002001D45005BA400A8009100AB0071008B008F009100920093009100000000000000000000000000AF00A500AD00A000B100A300AA00A200A3009C000000000000000000000000008A57E0B3AB0071008B008F009100920093009100000000000000000000000000AF00A500AD00A000B100A300AA00A200A3009C00000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A200000F070056A9000F416A9F62DF
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00001E080013EC00A2000133FB6E9B000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A200001F0700F40B005A625A6BF72B00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A00000100700E11E016450A25FACED
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA0000200800DE2100A000011D15EAC2000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A000002107006B94002900A6F807F200000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 13 13 0 B00200110500FA0500438F0629
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA000022080008F700B00201A289DE00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 128 0 B00200234500966900870088008800700085009600940070001A001A00000000000000000000000000AF00A500AD00A000B100A300AA00A300A3009D0000000000000000000000000074CA825A8800700085009600940070001A001A00000000000000000000000000AF00A500AD00A000B100A300AA00A300A3009D00000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A20000120700659A000F41892EC33C
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA0000240800758A00A20001D5DFAEE8000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A20000250700926D00556427B73E5900000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A000001307005CA3016450A3026C64
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA0000260800A35C00A00001F6560E07000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A0000027070016E90058265460564800000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 13 13 0 B002001405003AC500FE5F9E23
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00002808008F7000B00201DE4B8395000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 128 0 B0020029450011EE007B00950097006E0084008A007400720078008000000000000000000000000000AE00A500AD00A000B100A200AA00A300A3009D000000000000000000000000007AB44F7697006E0084008A007400720078008000000000000000000000000000AE00A500AD00A000B100A200AA00A300A3009D00000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A20000150700738C000F41F14DD4AC
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00002A080059A600A200010F033354000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A200002B0700BE4100566455244C8900000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A000001607009C630164501A0AD4B3
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00002C080024DB00A000018A945392000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A000002D0700916E005F2DD3D14D7200000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 13 13 0 B00200170500877800E690E739
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00002E0800F20D00B0020135086750000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 128 0 B002002F45006C9300A80083008A006D00870082008900940074001A00000000000000000000000000AF00A500AD00A000B100A300AA00A300A3009D00000000000000000000000000DD3369188A006D00870082008900940074001A00000000000000000000000000AF00A500AD00A000B100A300AA00A300A3009D00000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A20000180700E21D000F4141D04BDE
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00003008007C8300A2000111FF9D33000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A200003107009B6400586440DB796400000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A00000190700DB240164506BFCE486
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA0000320800AA5500A0000132763DDC000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A000003307001FE072635FE0E8A18200000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 13 13 0 B002001A050016E900A4EAF211
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA0000340800D72800B002012BF4C937000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 128 0 B0020035450049B6007D00920094008C009A008B008400890094006D00000000000000000000000000AE00A500AD00A000B100A300AA00A200A3009D00000000000000000000000000520A47F394008C009A008B008400890094006D00000000000000000000000000AE00A500AD00A000B100A300AA00A200A3009D00000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A200001B07005FA0000F41408D8B57
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA000036080001FE00A20001FABC79F6000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A20000370700E619005964B9BF1A2300000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A000001C07001BE4016450D2F45C51
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00003808002DD200A000014EB46049000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A000003907009867725F526A136B6D00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 13 13 0 B002001D050000FF0009B0C408
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00003A0800FB0400B00201F128548B000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 128 0 B002003B4500659A0084007D008B0099009C006A009A007B0089008300000000000000000000000000AF00A500AD00A000B100A300AA00A300A3009C00000000000000000000000000BAEEF1FA8B0099009C006A009A007B0089008300000000000000000000000000AF00A500AD00A000B100A300AA00A300A3009C00000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A200001E07009F60000F41F9853380
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00003C0800867900A20001867E2463000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A200003D0700619E005864307089D800000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A000001F0700A659016450D3A99CD8
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00003E080050AF00A00001A5F7848C000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A000003F0700E51A0063324BE2275A00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 13 13 0 B00200200500708F00DCB5EECB
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00004008001BE400B00201E7BD49D0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 128 0 B00200414500857A0078006F0074009200A6008C0071006C008E009400000000000000000000000000AF00A500AD00A000B100A300A900A200A3009C0000000000000000000000000079DB6FC674009200A6008C0071006C008E009400000000000000000000000000AF00A500AD00A000B100A300A900A200A3009C00000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A2000021070039C6000F41C7E8B94B
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA0000420800CD3200A2000136F5F911000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A200004307002AD5005A647427533000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A00000220700D6290164502CAFB954
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA0000440800B04F00A00001B36299D7000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A0000045070005FA0061447F5F4C1200000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 13 13 0 B00200230500CD3200C47A97D1
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA0000460800669900B002010CFEAD15000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 128 0 B00200474500F807009500860087008D00A3006D007D0081009C007C00000000000000000000000000AF00A500AD00A000B100A300AA00A300A3009C0000000000000000000000000074D7CC3787008D00A3006D007D0081009C007C00000000000000000000000000AF00A500AD00A000B100A300AA00A300A3009C00000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A20000240700F906000F417EE0019C
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00004808004AB500A200014A37A484000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A20000490700AD52005764F1A7756700000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A00000250700C03F01645054CCAEC4
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00004A08009C6300A0000169BE046B000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A000004B070029D60041281F96933800000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 13 13 0 B002002605000DF20079AA0FDB
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00004C0800E11E00B00201703CF080000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 128 0 B002004D45007F80009E008E008E007E00AB00840096007F0092008100000000000000000000000000AF00A500AD00A000B100A300AA00A300A3009C00000000000000000000000000D393E7738E007E00AB00840096007F0092008100000000000000000000000000AF00A500AD00A000B100A300AA00A300A3009C00000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A2000027070044BB000F417FBDC115
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00004E080037C800A20001A1744041000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A200004F0700D02F005964C7DF8EA700000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A0000028070051AE016450E45131B6
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA0000500800B94600A000017742AA0C000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A000005107000CF300624F1EC75A2600000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 13 13 0 B002002905004AB5002B5AB4E0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00005208006F9000B00201C8DE9ECE000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 128 0 B00200534500F10E0077007A00720083007500930073007E006E008200000000000000000000000000AE00A400AC009F00B100A200A900A200A3009C00000000000000000000000000C5190965720083007500930073007E006E008200000000000000000000000000AE00A400AC009F00B100A200A900A200A3009C00000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A200002A0700D52A000F41CF205E67
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA000054080012ED00A20001BF88EE26000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A20000550700F50A0058641D3C23CD00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A000002B0700EC13016450E50CF13F
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA0000560800C43B00A000019C014EC9000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A00000570700718E726054F685049700000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 13 13 0 B002002C05008A7500968A2CEA
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA0000580800E81700B00201B41CC35B000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 128 0 B002005945007689008C007100780072007A009500A10088009C008F00000000000000000000000000AE00A500AD00A000B100A200A900A200A3009C00000000000000000000000000D98A26F0780072007A009500A10088009C008F00000000000000000000000000AE00A500AD00A000B100A200A900A200A3009C00000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A200002D0700C33C000F41B74349F7
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00005A08003EC100A200016554739A000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A200005B0700D926005964EDCD672F00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A000002E07002CD30164505C0449E8
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00005C080043BC00A00001E0C3135C000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A000005D0700F609725F58291DE42400000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 13 13 0 B002002F050037C8008E4555F0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00005E0800956A00B002015F5F279E000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 128 0 B002005F45000BF400A30088009D008D00A80071007F0092006F006100000000000000000000000000AE00A400AC009F00B100A200A900A200A3009C000000000000000000000000001A1438ED9D008D00A80071007F0092006F006100000000000000000000000000AE00A400AC009F00B100A200A900A200A3009C00000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A20000300700F00F000F4154F2E814
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA000060080058A700A200018370B3E9000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A20000610700BF400059646AA8553300000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A00000310700C9360164507EDE474C
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00006208008E7100A00001A0F91306000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A000006307003BC4006432979DA1E700000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 13 13 0 B0020032050004FB00326A5B9D
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA0000640800F30C00B00201B97BE7ED000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 128 0 B002006545006D92008D008F008A00750083008D0092008B0071007A00000000000000000000000000AE00A500AD00A000B100A200A900A200A3009C00000000000000000000000000394AB8458A00750083008D0092008B0071007A00000000000000000000000000AE00A500AD00A000B100A200A900A200A3009C00000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A200003307004DB2000F4155AF289D
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA000066080025DA00A200016833572C000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A20000670700C23D0057645CD0AEF300000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A0000034070009F6016450C7D6FF9B
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA000068080009F600A00001DC3B4E93000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A00000690700BC43002E00509568C700000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A3000035070019E6000F41EC883513
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00006A0800DF2000A30001C8D82C7D000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A300006B05003BC40055C25B16000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 114 114 0 A40000366A006798640043010100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001E4650312D30303030303030302D302D30303030303030302D6E6F626F647900B5C8699F
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00006C0800A25D00A40001A68D87BD000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A400006D0500996600E41BAEE2000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 13 13 0 A60000370500689700E8D1ECD8
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00006E0800748B00A6000185042752000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 128 0 A600006F6C0055AA0001640043010107000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001E4650312D30303030303030302D302D30303030303030302D6E6F626F6479006EA4C362000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 A20000380700A15E010023D1112994
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA0000700800FA0500A20001E14E401B000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A200007107001DE2005A635942221700000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 44 44 0 A50000392400847B000000000000000000000000000000000000000000000000000000000000000038910386
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00007208002CD300A5000129052BF2000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 192 0 A50000739300A15E005A632900000000640043010107000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001E4650312D30303030303030302D302D30303030303030302D6E6F626F6479008B7767E4A81CC57C17A21C0F44B08232229E7F2E5DBFA70A5CEF58A3966ED710002FF9938030303030303030302D302D30303030303030302D6E6F626F6479008B7767E4A81CC57C17A2
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 15 15 0 a200003a07007788010023107a86d3
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA0000700800FA0500A20001E14E401B000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A200007107001DE2005A635942221700000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 44 44 0 a500003b240052ad0000000000000000000000000000000000000000000000000000000000000000828af704
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA00007208002CD300A5000129052BF2000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 192 0 A50000739300A15E005A632900000000640043010107000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001E4650312D30303030303030302D302D30303030303030302D6E6F626F6479008B7767E4A81CC57C17A21C0F44B08232229E7F2E5DBFA70A5CEF58A3966ED710002FF9938030303030303030302D302D30303030303030302D6E6F626F6479008B7767E4A81CC57C17A2
+USBDEVFS_REAPURBNDELAY 0 3 1 0 0 114 114 0 a700003c6a009b64640043010100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001e4650312d30303030303030302d302d30303030303030302d6e6f626f647900473c9b29
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 AA000074080051AE00A70001AC924B34000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 0 0
+ USBDEVFS_REAPURBNDELAY 0 3 131 0 0 2048 64 0 A7000075050011EE00ACFFDA28000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
diff --git a/tests/goodixmoc/custom.py b/tests/goodixmoc/custom.py
new file mode 100755
index 0000000..6cc3b70
--- /dev/null
+++ b/tests/goodixmoc/custom.py
@@ -0,0 +1,50 @@
+#!/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() == "goodixmoc"
+
+d.open_sync()
+
+template = FPrint.Print.new(d)
+
+def enroll_progress(*args):
+ print('enroll progress: ' + str(args))
+
+# List, enroll, list, verify, identify, delete
+print("enrolling")
+p = d.enroll_sync(template, None, enroll_progress, 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")
+verify_res, verify_print = d.verify_sync(p)
+print("verify done")
+assert verify_res == True
+print("identifying")
+identify_match, identify_print = d.identify_sync(stored)
+print("identify done")
+assert identify_match.equal(identify_print)
+
+print("deleting")
+d.delete_print_sync(p)
+print("delete done")
+d.close_sync()
+
+del d
+del c
diff --git a/tests/goodixmoc/device b/tests/goodixmoc/device
new file mode 100644
index 0000000..627c9dc
--- /dev/null
+++ b/tests/goodixmoc/device
@@ -0,0 +1,171 @@
+P: /devices/pci0000:00/0000:00:14.0/usb1/1-8
+N: bus/usb/001/003=12010002EF000040C627A26000010102030109022000010103A0320904000002FF0000040705830240000007050102400000
+E: DEVNAME=/dev/bus/usb/001/003
+E: DEVTYPE=usb_device
+E: DRIVER=usb
+E: PRODUCT=27c6/60a2/100
+E: TYPE=239/0/0
+E: BUSNUM=001
+E: DEVNUM=003
+E: MAJOR=189
+E: MINOR=2
+E: SUBSYSTEM=usb
+E: ID_VENDOR=Goodix_Technology_Co.__Ltd.
+E: ID_VENDOR_ENC=Goodix\x20Technology\x20Co.\x2c\x20Ltd.
+E: ID_VENDOR_ID=27c6
+E: ID_MODEL=Goodix_USB2.0_MISC
+E: ID_MODEL_ENC=Goodix\x20USB2.0\x20MISC
+E: ID_MODEL_ID=60a2
+E: ID_REVISION=0100
+E: ID_SERIAL=Goodix_Technology_Co.__Ltd._Goodix_USB2.0_MISC_UIDCBEE4D7B_XXXX_MOC_B0
+E: ID_SERIAL_SHORT=UIDCBEE4D7B_XXXX_MOC_B0
+E: ID_BUS=usb
+E: ID_USB_INTERFACES=:ff0000:
+E: ID_VENDOR_FROM_DATABASE=Shenzhen Goodix Technology Co.,Ltd.
+E: ID_PATH=pci-0000:00:14.0-usb-0:8
+E: ID_PATH_TAG=pci-0000_00_14_0-usb-0_8
+E: LIBFPRINT_DRIVER=AuthenTec AES1610
+A: authorized=1
+A: avoid_reset_quirk=0
+A: bConfigurationValue=1
+A: bDeviceClass=ef
+A: bDeviceProtocol=00
+A: bDeviceSubClass=00
+A: bMaxPacketSize0=64
+A: bMaxPower=100mA
+A: bNumConfigurations=1
+A: bNumInterfaces= 1
+A: bcdDevice=0100
+A: bmAttributes=a0
+A: busnum=1
+A: configuration=UIDCBEE4D7B_XXXX_MOC_B0
+H: descriptors=12010002EF000040C627A26000010102030109022000010103A0320904000002FF0000040705830240000007050102400000
+A: dev=189:2
+A: devnum=3
+A: devpath=8
+L: driver=../../../../../bus/usb/drivers/usb
+A: idProduct=60a2
+A: idVendor=27c6
+A: ltm_capable=no
+A: manufacturer=Goodix Technology Co., Ltd.
+A: maxchild=0
+L: port=../1-0:1.0/usb1-port8
+A: power/active_duration=324448
+A: power/async=enabled
+A: power/autosuspend=2
+A: power/autosuspend_delay_ms=2000
+A: power/connected_duration=5916532
+A: power/control=auto
+A: power/level=auto
+A: power/persist=1
+A: power/runtime_active_kids=0
+A: power/runtime_active_time=327268
+A: power/runtime_enabled=enabled
+A: power/runtime_status=active
+A: power/runtime_suspended_time=5588987
+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=Goodix USB2.0 MISC
+A: quirks=0x0
+A: removable=fixed
+A: rx_lanes=1
+A: serial=UIDCBEE4D7B_XXXX_MOC_B0
+A: speed=12
+A: tx_lanes=1
+A: urbnum=2180
+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-29-generic_xhci-hcd
+E: ID_VENDOR_ENC=Linux\x205.4.0-29-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-29-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-29-generic xhci-hcd
+A: maxchild=12
+A: power/active_duration=5879432
+A: power/async=enabled
+A: power/autosuspend=0
+A: power/autosuspend_delay_ms=0
+A: power/connected_duration=5916912
+A: power/control=auto
+A: power/level=auto
+A: power/runtime_active_kids=2
+A: power/runtime_active_time=5879430
+A: power/runtime_enabled=enabled
+A: power/runtime_status=active
+A: power/runtime_suspended_time=37481
+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=1319
+A: version= 2.00
diff --git a/tests/meson.build b/tests/meson.build
index c2ba04c..70044b8 100644
--- a/tests/meson.build
+++ b/tests/meson.build
@@ -21,6 +21,7 @@ drivers_tests = [
'synaptics',
'vfs0050',
'vfs5011',
+ 'goodixmoc',
]
if get_option('introspection')