Line | Branch | Exec | Source |
---|---|---|---|
1 | /* | ||
2 | * Virtual driver for "simple" device debugging | ||
3 | * | ||
4 | * Copyright (C) 2019 Benjamin Berg <bberg@redhat.com> | ||
5 | * Copyright (C) 2020 Bastien Nocera <hadess@hadess.net> | ||
6 | * Copyright (C) 2020 Marco Trevisan <marco.trevisan@canonical.com> | ||
7 | * | ||
8 | * This library is free software; you can redistribute it and/or | ||
9 | * modify it under the terms of the GNU Lesser General Public | ||
10 | * License as published by the Free Software Foundation; either | ||
11 | * version 2.1 of the License, or (at your option) any later version. | ||
12 | * | ||
13 | * This library is distributed in the hope that it will be useful, | ||
14 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
15 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | ||
16 | * Lesser General Public License for more details. | ||
17 | * | ||
18 | * You should have received a copy of the GNU Lesser General Public | ||
19 | * License along with this library; if not, write to the Free Software | ||
20 | * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA | ||
21 | */ | ||
22 | |||
23 | /* | ||
24 | * This is a virtual driver to debug the non-image based drivers. A small | ||
25 | * python script is provided to connect to it via a socket, allowing | ||
26 | * prints to registered programmatically. | ||
27 | * Using this, it is possible to test libfprint and fprintd. | ||
28 | */ | ||
29 | |||
30 | #define FP_COMPONENT "virtual_device" | ||
31 | |||
32 | #include "virtual-device-private.h" | ||
33 | #include "fpi-log.h" | ||
34 | |||
35 |
4/5✓ Branch 0 taken 120 times.
✓ Branch 1 taken 144 times.
✓ Branch 2 taken 120 times.
✓ Branch 3 taken 120 times.
✗ Branch 4 not taken.
|
1008 | G_DEFINE_TYPE (FpDeviceVirtualDevice, fpi_device_virtual_device, FP_TYPE_DEVICE) |
36 | |||
37 | #define INSERT_CMD_PREFIX "INSERT " | ||
38 | #define REMOVE_CMD_PREFIX "REMOVE " | ||
39 | #define SCAN_CMD_PREFIX "SCAN " | ||
40 | #define CONT_CMD_PREFIX "CONT " | ||
41 | #define ERROR_CMD_PREFIX "ERROR " | ||
42 | #define RETRY_CMD_PREFIX "RETRY " | ||
43 | #define FINGER_CMD_PREFIX "FINGER " | ||
44 | #define SLEEP_CMD_PREFIX "SLEEP " | ||
45 | #define SET_ENROLL_STAGES_PREFIX "SET_ENROLL_STAGES " | ||
46 | #define SET_SCAN_TYPE_PREFIX "SET_SCAN_TYPE " | ||
47 | #define SET_CANCELLATION_PREFIX "SET_CANCELLATION_ENABLED " | ||
48 | #define SET_KEEP_ALIVE_PREFIX "SET_KEEP_ALIVE " | ||
49 | |||
50 | #define LIST_CMD "LIST" | ||
51 | #define UNPLUG_CMD "UNPLUG" | ||
52 | |||
53 | static void | ||
54 | 522 | maybe_continue_current_action (FpDeviceVirtualDevice *self) | |
55 | { | ||
56 | 522 | FpDevice *dev = FP_DEVICE (self); | |
57 | |||
58 |
2/2✓ Branch 0 taken 518 times.
✓ Branch 1 taken 4 times.
|
522 | if (self->sleep_timeout_id) |
59 | return; | ||
60 | |||
61 |
1/2✓ Branch 0 taken 518 times.
✗ Branch 1 not taken.
|
518 | g_assert (self->wait_command_id == 0); |
62 | |||
63 |
8/9✓ Branch 1 taken 244 times.
✓ Branch 2 taken 14 times.
✓ Branch 3 taken 11 times.
✓ Branch 4 taken 8 times.
✓ Branch 5 taken 5 times.
✓ Branch 6 taken 7 times.
✓ Branch 7 taken 10 times.
✗ Branch 8 not taken.
✓ Branch 9 taken 219 times.
|
518 | switch (fpi_device_get_current_action (dev)) |
64 | { | ||
65 | case FPI_DEVICE_ACTION_ENROLL: | ||
66 | 244 | FP_DEVICE_GET_CLASS (self)->enroll (dev); | |
67 | 244 | break; | |
68 | |||
69 | case FPI_DEVICE_ACTION_VERIFY: | ||
70 | 14 | FP_DEVICE_GET_CLASS (self)->verify (dev); | |
71 | 14 | break; | |
72 | |||
73 | case FPI_DEVICE_ACTION_IDENTIFY: | ||
74 | 11 | FP_DEVICE_GET_CLASS (self)->identify (dev); | |
75 | 11 | break; | |
76 | |||
77 | case FPI_DEVICE_ACTION_LIST: | ||
78 | 8 | FP_DEVICE_GET_CLASS (self)->list (dev); | |
79 | 8 | break; | |
80 | |||
81 | case FPI_DEVICE_ACTION_DELETE: | ||
82 | 5 | FP_DEVICE_GET_CLASS (self)->delete (dev); | |
83 | 5 | break; | |
84 | |||
85 | case FPI_DEVICE_ACTION_OPEN: | ||
86 | 7 | FP_DEVICE_GET_CLASS (self)->open (dev); | |
87 | 7 | break; | |
88 | |||
89 | case FPI_DEVICE_ACTION_CLOSE: | ||
90 | 10 | FP_DEVICE_GET_CLASS (self)->close (dev); | |
91 | 10 | break; | |
92 | |||
93 | case FPI_DEVICE_ACTION_CLEAR_STORAGE: | ||
94 | ✗ | FP_DEVICE_GET_CLASS (self)->clear_storage (dev); | |
95 | ✗ | break; | |
96 | |||
97 | /* Not implemented/nothing to do. */ | ||
98 | case FPI_DEVICE_ACTION_NONE: | ||
99 | case FPI_DEVICE_ACTION_PROBE: | ||
100 | case FPI_DEVICE_ACTION_CAPTURE: | ||
101 | default: | ||
102 | break; | ||
103 | } | ||
104 | } | ||
105 | |||
106 | static gboolean | ||
107 | 187 | sleep_timeout_cb (gpointer data) | |
108 | { | ||
109 | 187 | FpDeviceVirtualDevice *self = data; | |
110 | |||
111 | 187 | self->sleep_timeout_id = 0; | |
112 | |||
113 |
1/2✓ Branch 1 taken 187 times.
✗ Branch 2 not taken.
|
187 | if (g_cancellable_is_cancelled (self->cancellable)) |
114 | return FALSE; | ||
115 | |||
116 | 187 | g_debug ("Sleeping completed"); | |
117 | 187 | maybe_continue_current_action (self); | |
118 | |||
119 | 187 | return FALSE; | |
120 | } | ||
121 | |||
122 | static gboolean | ||
123 | 12 | wait_for_command_timeout (gpointer data) | |
124 | { | ||
125 | 12 | FpDeviceVirtualDevice *self = FP_DEVICE_VIRTUAL_DEVICE (data); | |
126 | 12 | FpiDeviceAction action; | |
127 | 12 | GError *error = NULL; | |
128 | |||
129 | 12 | self->wait_command_id = 0; | |
130 | |||
131 | 12 | action = fpi_device_get_current_action (FP_DEVICE (self)); | |
132 |
2/2✓ Branch 0 taken 11 times.
✓ Branch 1 taken 1 times.
|
12 | if (action == FPI_DEVICE_ACTION_LIST || action == FPI_DEVICE_ACTION_DELETE) |
133 | { | ||
134 | 11 | self->ignore_wait = TRUE; | |
135 | 11 | maybe_continue_current_action (self); | |
136 | 11 | self->ignore_wait = FALSE; | |
137 | |||
138 | 11 | return FALSE; | |
139 | } | ||
140 | |||
141 | 1 | error = g_error_new (G_IO_ERROR, G_IO_ERROR_TIMED_OUT, "No commands arrived in time to run!"); | |
142 | 1 | fpi_device_action_error (FP_DEVICE (self), error); | |
143 | |||
144 | 1 | return FALSE; | |
145 | } | ||
146 | |||
147 | gboolean | ||
148 | 614 | process_cmds (FpDeviceVirtualDevice * self, | |
149 | gboolean scan, | ||
150 | char **scan_id, | ||
151 | GError **error) | ||
152 | { | ||
153 | 614 | gboolean removed; | |
154 | |||
155 |
2/4✓ Branch 1 taken 614 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 614 times.
✗ Branch 4 not taken.
|
1228 | if (g_cancellable_is_cancelled (self->cancellable) || |
156 |
2/2✓ Branch 1 taken 606 times.
✓ Branch 2 taken 8 times.
|
1228 | (fpi_device_get_current_action (FP_DEVICE (self)) != FPI_DEVICE_ACTION_NONE && |
157 | 614 | g_cancellable_is_cancelled (fpi_device_get_cancellable (FP_DEVICE (self))))) | |
158 | { | ||
159 | 8 | g_set_error_literal (error, G_IO_ERROR, G_IO_ERROR_CANCELLED, | |
160 | "Operation was cancelled"); | ||
161 | 8 | return TRUE; | |
162 | } | ||
163 | |||
164 |
2/2✓ Branch 0 taken 318 times.
✓ Branch 1 taken 300 times.
|
618 | while (self->pending_commands->len > 0) |
165 | { | ||
166 | 318 | g_autofree gchar *cmd = NULL; | |
167 | |||
168 | /* TODO: g_ptr_array_steal_index requires GLib 2.58, we depend on 2.56 */ | ||
169 | 318 | cmd = g_ptr_array_index (self->pending_commands, 0); | |
170 | 318 | g_ptr_array_index (self->pending_commands, 0) = NULL; | |
171 | 318 | g_ptr_array_remove_index (self->pending_commands, 0); | |
172 | |||
173 | 318 | g_debug ("Processing command %s", cmd); | |
174 | |||
175 | /* These are always processed. */ | ||
176 |
5/6✗ Branch 0 not taken.
✓ Branch 1 taken 318 times.
✓ Branch 3 taken 279 times.
✓ Branch 4 taken 39 times.
✓ Branch 5 taken 3 times.
✓ Branch 6 taken 276 times.
|
318 | if (g_str_has_prefix (cmd, INSERT_CMD_PREFIX)) |
177 | { | ||
178 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 3 times.
|
3 | g_assert (self->prints_storage); |
179 | 6 | g_hash_table_add (self->prints_storage, | |
180 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 3 times.
|
6 | g_strdup (cmd + strlen (INSERT_CMD_PREFIX))); |
181 | |||
182 | 3 | continue; | |
183 | } | ||
184 |
5/6✓ Branch 0 taken 315 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 276 times.
✓ Branch 3 taken 39 times.
✓ Branch 5 taken 1 times.
✓ Branch 6 taken 275 times.
|
315 | else if (g_str_has_prefix (cmd, REMOVE_CMD_PREFIX)) |
185 | { | ||
186 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
|
1 | g_assert (self->prints_storage); |
187 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 1 times.
|
1 | if (!g_hash_table_remove (self->prints_storage, |
188 | 1 | cmd + strlen (REMOVE_CMD_PREFIX))) | |
189 | ✗ | g_warning ("ID %s was not found in storage", cmd + strlen (REMOVE_CMD_PREFIX)); | |
190 | |||
191 | 1 | continue; | |
192 | } | ||
193 |
5/6✓ Branch 0 taken 314 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 275 times.
✓ Branch 3 taken 39 times.
✓ Branch 5 taken 43 times.
✓ Branch 6 taken 232 times.
|
314 | else if (g_str_has_prefix (cmd, SLEEP_CMD_PREFIX)) |
194 | { | ||
195 | 43 | guint64 sleep_ms = g_ascii_strtoull (cmd + strlen (SLEEP_CMD_PREFIX), NULL, 10); | |
196 | |||
197 | 43 | g_debug ("Sleeping %" G_GUINT64_FORMAT "ms", sleep_ms); | |
198 | 43 | self->sleep_timeout_id = g_timeout_add (sleep_ms, sleep_timeout_cb, self); | |
199 | |||
200 | 43 | return FALSE; | |
201 | } | ||
202 |
5/6✓ Branch 0 taken 271 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 232 times.
✓ Branch 3 taken 39 times.
✓ Branch 5 taken 11 times.
✓ Branch 6 taken 221 times.
|
271 | else if (g_str_has_prefix (cmd, ERROR_CMD_PREFIX)) |
203 | { | ||
204 | 11 | g_propagate_error (error, | |
205 | 11 | fpi_device_error_new (g_ascii_strtoull (cmd + strlen (ERROR_CMD_PREFIX), NULL, 10))); | |
206 | |||
207 | 11 | return TRUE; | |
208 | } | ||
209 |
6/8✓ Branch 0 taken 45 times.
✓ Branch 1 taken 215 times.
✓ Branch 2 taken 45 times.
✗ Branch 3 not taken.
✓ Branch 4 taken 45 times.
✗ Branch 5 not taken.
✓ Branch 7 taken 6 times.
✓ Branch 8 taken 39 times.
|
260 | else if (!scan && g_str_has_prefix (cmd, CONT_CMD_PREFIX)) |
210 | { | ||
211 | return TRUE; | ||
212 | } | ||
213 | |||
214 | /* If we are not scanning, then we have to stop here. */ | ||
215 | 221 | if (!scan) | |
216 | { | ||
217 | 6 | g_warning ("Could not process command: %s", cmd); | |
218 | 6 | break; | |
219 | } | ||
220 | |||
221 |
4/6✓ Branch 0 taken 215 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 215 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 174 times.
✓ Branch 6 taken 41 times.
|
215 | if (g_str_has_prefix (cmd, SCAN_CMD_PREFIX)) |
222 | { | ||
223 |
1/2✓ Branch 0 taken 174 times.
✗ Branch 1 not taken.
|
174 | if (scan_id) |
224 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 174 times.
|
348 | *scan_id = g_strdup (cmd + strlen (SCAN_CMD_PREFIX)); |
225 | |||
226 | 174 | return TRUE; | |
227 | } | ||
228 |
4/6✓ Branch 0 taken 41 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 41 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 33 times.
✓ Branch 6 taken 8 times.
|
41 | else if (g_str_has_prefix (cmd, RETRY_CMD_PREFIX)) |
229 | { | ||
230 | 33 | g_propagate_error (error, | |
231 | 33 | fpi_device_retry_new (g_ascii_strtoull (cmd + strlen (RETRY_CMD_PREFIX), NULL, 10))); | |
232 | |||
233 | 33 | return TRUE; | |
234 | } | ||
235 |
3/6✓ Branch 0 taken 8 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 8 times.
✗ Branch 3 not taken.
✓ Branch 5 taken 8 times.
✗ Branch 6 not taken.
|
8 | else if (g_str_has_prefix (cmd, FINGER_CMD_PREFIX)) |
236 | 8 | { | |
237 | 8 | gboolean finger_present; | |
238 | |||
239 | 8 | finger_present = g_ascii_strtoull (cmd + strlen (FINGER_CMD_PREFIX), NULL, 10) != 0; | |
240 |
2/2✓ Branch 0 taken 4 times.
✓ Branch 1 taken 4 times.
|
12 | fpi_device_report_finger_status_changes (FP_DEVICE (self), |
241 | finger_present ? FP_FINGER_STATUS_PRESENT : FP_FINGER_STATUS_NONE, | ||
242 | finger_present ? FP_FINGER_STATUS_NONE : FP_FINGER_STATUS_PRESENT); | ||
243 | |||
244 | 8 | continue; | |
245 | } | ||
246 | else | ||
247 | { | ||
248 | ✗ | g_warning ("Could not process command: %s", cmd); | |
249 | } | ||
250 | } | ||
251 | |||
252 |
2/2✓ Branch 0 taken 195 times.
✓ Branch 1 taken 111 times.
|
306 | if (self->ignore_wait) |
253 | return TRUE; | ||
254 | |||
255 | 111 | g_object_get (self, "removed", &removed, NULL); | |
256 | |||
257 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 111 times.
|
111 | g_assert (self->wait_command_id == 0); |
258 |
4/4✓ Branch 0 taken 100 times.
✓ Branch 1 taken 11 times.
✓ Branch 2 taken 1 times.
✓ Branch 3 taken 99 times.
|
111 | if (!scan || removed) |
259 | 12 | self->wait_command_id = g_timeout_add (500, wait_for_command_timeout, self); | |
260 | return FALSE; | ||
261 | } | ||
262 | |||
263 | static void | ||
264 | ✗ | write_key_to_listener (void *key, void *val, void *user_data) | |
265 | { | ||
266 | ✗ | FpiDeviceVirtualListener *listener = FPI_DEVICE_VIRTUAL_LISTENER (user_data); | |
267 | |||
268 | ✗ | if (!fpi_device_virtual_listener_write_sync (listener, key, strlen (key), NULL) || | |
269 | ✗ | !fpi_device_virtual_listener_write_sync (listener, "\n", 1, NULL)) | |
270 | ✗ | g_warning ("Error writing reply to LIST command"); | |
271 | ✗ | } | |
272 | |||
273 | static void | ||
274 | 467 | recv_instruction_cb (GObject *source_object, | |
275 | GAsyncResult *res, | ||
276 | gpointer user_data) | ||
277 | { | ||
278 | 467 | g_autoptr(GError) error = NULL; | |
279 | 467 | FpiDeviceVirtualListener *listener = FPI_DEVICE_VIRTUAL_LISTENER (source_object); | |
280 | 467 | gsize bytes; | |
281 | |||
282 | 467 | bytes = fpi_device_virtual_listener_read_finish (listener, res, &error); | |
283 | 467 | fp_dbg ("Got instructions of length %" G_GSIZE_FORMAT, bytes); | |
284 | |||
285 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 467 times.
|
467 | if (error) |
286 | { | ||
287 | ✗ | if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED)) | |
288 | return; | ||
289 | |||
290 | ✗ | g_warning ("Error receiving instruction data: %s", error->message); | |
291 | ✗ | return; | |
292 | } | ||
293 | |||
294 |
1/2✓ Branch 0 taken 467 times.
✗ Branch 1 not taken.
|
467 | if (bytes > 0) |
295 | { | ||
296 | 467 | FpDeviceVirtualDevice *self; | |
297 | 467 | g_autofree char *cmd = NULL; | |
298 | |||
299 | 467 | self = FP_DEVICE_VIRTUAL_DEVICE (user_data); | |
300 | |||
301 | 467 | cmd = g_strndup (self->recv_buf, bytes); | |
302 | 467 | fp_dbg ("Received command %s", cmd); | |
303 | |||
304 |
3/6✗ Branch 0 not taken.
✓ Branch 1 taken 467 times.
✓ Branch 3 taken 467 times.
✗ Branch 4 not taken.
✗ Branch 5 not taken.
✓ Branch 6 taken 467 times.
|
467 | if (g_str_has_prefix (cmd, LIST_CMD)) |
305 | { | ||
306 | ✗ | if (self->prints_storage) | |
307 | ✗ | g_hash_table_foreach (self->prints_storage, write_key_to_listener, listener); | |
308 | } | ||
309 |
5/6✓ Branch 0 taken 467 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 428 times.
✓ Branch 3 taken 39 times.
✓ Branch 5 taken 2 times.
✓ Branch 6 taken 426 times.
|
467 | else if (g_str_has_prefix (cmd, UNPLUG_CMD)) |
310 | { | ||
311 | 2 | fpi_device_remove (FP_DEVICE (self)); | |
312 | 2 | maybe_continue_current_action (self); | |
313 | } | ||
314 |
5/6✓ Branch 0 taken 465 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 85 times.
✓ Branch 3 taken 380 times.
✓ Branch 5 taken 73 times.
✓ Branch 6 taken 12 times.
|
465 | else if (g_str_has_prefix (cmd, SET_ENROLL_STAGES_PREFIX)) |
315 | { | ||
316 | 73 | guint stages; | |
317 | |||
318 | 73 | stages = g_ascii_strtoull (cmd + strlen (SET_ENROLL_STAGES_PREFIX), NULL, 10); | |
319 | 73 | fpi_device_set_nr_enroll_stages (FP_DEVICE (self), stages); | |
320 | } | ||
321 |
5/6✓ Branch 0 taken 392 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 206 times.
✓ Branch 3 taken 186 times.
✓ Branch 5 taken 6 times.
✓ Branch 6 taken 200 times.
|
392 | else if (g_str_has_prefix (cmd, SET_SCAN_TYPE_PREFIX)) |
322 | { | ||
323 | 6 | const char *scan_type = cmd + strlen (SET_SCAN_TYPE_PREFIX); | |
324 | 473 | g_autoptr(GEnumClass) scan_types = g_type_class_ref (fp_scan_type_get_type ()); | |
325 | 6 | GEnumValue *value = g_enum_get_value_by_nick (scan_types, scan_type); | |
326 | |||
327 |
2/2✓ Branch 0 taken 4 times.
✓ Branch 1 taken 2 times.
|
6 | if (value) |
328 | 4 | fpi_device_set_scan_type (FP_DEVICE (self), value->value); | |
329 | else | ||
330 | 2 | g_warning ("Scan type '%s' not found", scan_type); | |
331 | } | ||
332 |
4/6✓ Branch 0 taken 386 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 2 times.
✓ Branch 3 taken 384 times.
✓ Branch 5 taken 2 times.
✗ Branch 6 not taken.
|
386 | else if (g_str_has_prefix (cmd, SET_CANCELLATION_PREFIX)) |
333 | { | ||
334 | 4 | self->supports_cancellation = g_ascii_strtoull ( | |
335 | 2 | cmd + strlen (SET_CANCELLATION_PREFIX), NULL, 10) != 0; | |
336 | |||
337 | 2 | g_debug ("Cancellation support toggled: %d", | |
338 | self->supports_cancellation); | ||
339 | } | ||
340 |
5/6✓ Branch 0 taken 384 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 114 times.
✓ Branch 3 taken 270 times.
✓ Branch 5 taken 68 times.
✓ Branch 6 taken 46 times.
|
384 | else if (g_str_has_prefix (cmd, SET_KEEP_ALIVE_PREFIX)) |
341 | { | ||
342 | 136 | self->keep_alive = g_ascii_strtoull ( | |
343 | 68 | cmd + strlen (SET_KEEP_ALIVE_PREFIX), NULL, 10) != 0; | |
344 | |||
345 | 68 | g_debug ("Keep alive toggled: %d", self->keep_alive); | |
346 | } | ||
347 | else | ||
348 | { | ||
349 | 316 | g_ptr_array_add (self->pending_commands, g_steal_pointer (&cmd)); | |
350 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 316 times.
|
316 | g_clear_handle_id (&self->wait_command_id, g_source_remove); |
351 | |||
352 | 316 | maybe_continue_current_action (self); | |
353 | } | ||
354 | } | ||
355 | |||
356 |
1/2✗ Branch 1 not taken.
✓ Branch 2 taken 467 times.
|
467 | fpi_device_virtual_listener_connection_close (listener); |
357 | } | ||
358 | |||
359 | static void | ||
360 | 467 | recv_instruction (FpDeviceVirtualDevice *self) | |
361 | { | ||
362 | 467 | fpi_device_virtual_listener_read (self->listener, | |
363 | FALSE, | ||
364 | 467 | self->recv_buf, | |
365 | sizeof (self->recv_buf), | ||
366 | recv_instruction_cb, | ||
367 | self); | ||
368 | 467 | } | |
369 | |||
370 | static void | ||
371 | 467 | on_listener_connected (FpiDeviceVirtualListener *listener, | |
372 | gpointer user_data) | ||
373 | { | ||
374 | 467 | FpDeviceVirtualDevice *self = FP_DEVICE_VIRTUAL_DEVICE (user_data); | |
375 | |||
376 | 467 | recv_instruction (self); | |
377 | 467 | } | |
378 | |||
379 | static void | ||
380 | 104 | dev_init (FpDevice *dev) | |
381 | { | ||
382 | 104 | g_autoptr(GError) error = NULL; | |
383 |
2/4✗ Branch 0 not taken.
✓ Branch 1 taken 88 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 16 times.
|
104 | g_autoptr(GCancellable) cancellable = NULL; |
384 |
2/4✗ Branch 0 not taken.
✓ Branch 1 taken 88 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 16 times.
|
104 | g_autoptr(FpiDeviceVirtualListener) listener = NULL; |
385 | 104 | FpDeviceVirtualDevice *self = FP_DEVICE_VIRTUAL_DEVICE (dev); | |
386 | |||
387 | 104 | G_DEBUG_HERE (); | |
388 | |||
389 | 104 | self->ignore_wait = TRUE; | |
390 |
2/2✓ Branch 1 taken 7 times.
✓ Branch 2 taken 97 times.
|
104 | if (!process_cmds (self, FALSE, NULL, &error)) |
391 | { | ||
392 | 7 | self->ignore_wait = FALSE; | |
393 | 7 | return; | |
394 | } | ||
395 | 97 | self->ignore_wait = FALSE; | |
396 | |||
397 |
2/2✓ Branch 0 taken 4 times.
✓ Branch 1 taken 93 times.
|
97 | if (error) |
398 | { | ||
399 | 4 | fpi_device_open_complete (dev, g_steal_pointer (&error)); | |
400 | 4 | return; | |
401 | } | ||
402 |
2/2✓ Branch 0 taken 5 times.
✓ Branch 1 taken 88 times.
|
93 | else if (self->listener) |
403 | { | ||
404 | 5 | fpi_device_open_complete (dev, NULL); | |
405 | 5 | return; | |
406 | } | ||
407 | |||
408 | 88 | listener = fpi_device_virtual_listener_new (); | |
409 | 88 | cancellable = g_cancellable_new (); | |
410 | |||
411 |
1/2✗ Branch 1 not taken.
✓ Branch 2 taken 88 times.
|
88 | if (!fpi_device_virtual_listener_start (listener, |
412 | 88 | fpi_device_get_virtual_env (FP_DEVICE (self)), | |
413 | cancellable, | ||
414 | on_listener_connected, | ||
415 | self, | ||
416 | &error)) | ||
417 | { | ||
418 | ✗ | fpi_device_open_complete (dev, g_steal_pointer (&error)); | |
419 | ✗ | return; | |
420 | } | ||
421 | |||
422 | 88 | self->listener = g_steal_pointer (&listener); | |
423 | 88 | self->cancellable = g_steal_pointer (&cancellable); | |
424 | |||
425 |
1/2✗ Branch 1 not taken.
✓ Branch 2 taken 88 times.
|
88 | fpi_device_open_complete (dev, NULL); |
426 | } | ||
427 | |||
428 | gboolean | ||
429 | 333 | start_scan_command (FpDeviceVirtualDevice *self, | |
430 | char **scan_id, | ||
431 | GError **error) | ||
432 | { | ||
433 | 666 | g_autoptr(GError) local_error = NULL; | |
434 | 333 | gboolean cont; | |
435 | |||
436 |
2/2✓ Branch 1 taken 76 times.
✓ Branch 2 taken 257 times.
|
333 | if (fp_device_get_finger_status (FP_DEVICE (self)) == FP_FINGER_STATUS_NONE) |
437 | 76 | self->injected_synthetic_cmd = FALSE; | |
438 | |||
439 | 333 | cont = process_cmds (self, TRUE, scan_id, &local_error); | |
440 | /* We report finger needed if we are waiting for instructions | ||
441 | * (i.e. we did not get an explicit SLEEP command). | ||
442 | */ | ||
443 |
2/2✓ Branch 0 taken 317 times.
✓ Branch 1 taken 16 times.
|
333 | if (!self->sleep_timeout_id) |
444 | { | ||
445 | 317 | fpi_device_report_finger_status_changes (FP_DEVICE (self), | |
446 | FP_FINGER_STATUS_NEEDED, | ||
447 | FP_FINGER_STATUS_NONE); | ||
448 | } | ||
449 | |||
450 |
2/2✓ Branch 0 taken 217 times.
✓ Branch 1 taken 116 times.
|
333 | if (!cont) |
451 | return FALSE; | ||
452 | |||
453 | /* Scan or error*/ | ||
454 | 217 | fpi_device_report_finger_status_changes (FP_DEVICE (self), | |
455 | FP_FINGER_STATUS_NEEDED, | ||
456 | FP_FINGER_STATUS_NONE); | ||
457 | |||
458 |
2/2✓ Branch 0 taken 43 times.
✓ Branch 1 taken 174 times.
|
217 | if (local_error) |
459 | 43 | g_propagate_error (error, g_steal_pointer (&local_error)); | |
460 | else | ||
461 | 174 | fpi_device_report_finger_status_changes (FP_DEVICE (self), | |
462 | FP_FINGER_STATUS_PRESENT, | ||
463 | FP_FINGER_STATUS_NONE); | ||
464 | |||
465 | return TRUE; | ||
466 | } | ||
467 | |||
468 | gboolean | ||
469 | 188 | should_wait_to_sleep (FpDeviceVirtualDevice *self, | |
470 | const char *scan_id, | ||
471 | GError *error) | ||
472 | { | ||
473 | 188 | const gchar *cmd; | |
474 | |||
475 |
1/2✓ Branch 0 taken 188 times.
✗ Branch 1 not taken.
|
188 | if (self->sleep_timeout_id) |
476 | return TRUE; | ||
477 | |||
478 |
2/2✓ Branch 0 taken 70 times.
✓ Branch 1 taken 118 times.
|
188 | if (!self->pending_commands->len) |
479 | return FALSE; | ||
480 | |||
481 | 70 | cmd = g_ptr_array_index (self->pending_commands, 0); | |
482 | |||
483 |
4/6✗ Branch 0 not taken.
✓ Branch 1 taken 70 times.
✓ Branch 3 taken 70 times.
✗ Branch 4 not taken.
✓ Branch 5 taken 8 times.
✓ Branch 6 taken 62 times.
|
70 | if (g_str_has_prefix (cmd, SLEEP_CMD_PREFIX)) |
484 | { | ||
485 | 8 | g_autoptr(GError) local_error = NULL; | |
486 | 8 | process_cmds (self, FALSE, NULL, &local_error); | |
487 | |||
488 |
1/2✓ Branch 2 taken 8 times.
✗ Branch 3 not taken.
|
8 | if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED)) |
489 | return FALSE; | ||
490 | |||
491 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 8 times.
|
8 | g_assert (!self->injected_synthetic_cmd); |
492 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 8 times.
|
8 | g_assert (self->sleep_timeout_id != 0); |
493 | |||
494 |
2/2✓ Branch 0 taken 2 times.
✓ Branch 1 taken 6 times.
|
8 | if (!self->pending_commands->len) |
495 | { | ||
496 | 2 | g_autofree char *injected_cmd = NULL; | |
497 | |||
498 |
1/2✓ Branch 0 taken 2 times.
✗ Branch 1 not taken.
|
2 | if (scan_id) |
499 | 2 | injected_cmd = g_strconcat (SCAN_CMD_PREFIX, scan_id, NULL); | |
500 | ✗ | else if (error && error->domain == FP_DEVICE_ERROR) | |
501 | ✗ | injected_cmd = g_strdup_printf (ERROR_CMD_PREFIX " %d", error->code); | |
502 | ✗ | else if (error && error->domain == FP_DEVICE_RETRY) | |
503 | ✗ | injected_cmd = g_strdup_printf (RETRY_CMD_PREFIX " %d", error->code); | |
504 | else | ||
505 | ✗ | return TRUE; | |
506 | |||
507 | 2 | g_debug ("Sleeping now, command queued for later: %s", injected_cmd); | |
508 | |||
509 | 2 | g_ptr_array_insert (self->pending_commands, 0, g_steal_pointer (&injected_cmd)); | |
510 | 2 | self->injected_synthetic_cmd = TRUE; | |
511 | } | ||
512 | } | ||
513 | |||
514 | 70 | return self->sleep_timeout_id != 0; | |
515 | } | ||
516 | |||
517 | static void | ||
518 | 34 | dev_verify (FpDevice *dev) | |
519 | { | ||
520 | 34 | g_autoptr(GError) error = NULL; | |
521 | 34 | FpDeviceVirtualDevice *self = FP_DEVICE_VIRTUAL_DEVICE (dev); | |
522 |
2/4✗ Branch 0 not taken.
✓ Branch 1 taken 19 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 15 times.
|
34 | g_autofree char *scan_id = NULL; |
523 | |||
524 |
2/2✓ Branch 1 taken 20 times.
✓ Branch 2 taken 14 times.
|
34 | if (!start_scan_command (self, &scan_id, &error)) |
525 | return; | ||
526 | |||
527 |
2/2✓ Branch 0 taken 9 times.
✓ Branch 1 taken 11 times.
|
20 | if (scan_id) |
528 | { | ||
529 | 9 | g_autoptr(FpPrint) new_scan = NULL; | |
530 | 9 | GVariant *data = NULL; | |
531 | 9 | FpPrint *print; | |
532 | 9 | gboolean success; | |
533 | |||
534 | 9 | g_debug ("Virtual device scanned print %s", scan_id); | |
535 | 9 | fpi_device_get_verify_data (dev, &print); | |
536 | |||
537 | 9 | new_scan = fp_print_new (dev); | |
538 | 9 | fpi_print_set_type (new_scan, FPI_PRINT_RAW); | |
539 |
2/2✓ Branch 0 taken 3 times.
✓ Branch 1 taken 6 times.
|
9 | if (self->prints_storage) |
540 | 3 | fpi_print_set_device_stored (new_scan, TRUE); | |
541 | 9 | data = g_variant_new_string (scan_id); | |
542 | 9 | g_object_set (new_scan, "fpi-data", data, NULL); | |
543 | |||
544 |
4/4✓ Branch 0 taken 3 times.
✓ Branch 1 taken 6 times.
✓ Branch 3 taken 2 times.
✓ Branch 4 taken 1 times.
|
9 | if (self->prints_storage && !g_hash_table_contains (self->prints_storage, scan_id)) |
545 | { | ||
546 |
1/2✓ Branch 0 taken 2 times.
✗ Branch 1 not taken.
|
2 | g_clear_object (&new_scan); |
547 | success = FALSE; | ||
548 | } | ||
549 | else | ||
550 | { | ||
551 | 7 | success = fp_print_equal (print, new_scan); | |
552 | } | ||
553 | |||
554 |
2/2✓ Branch 0 taken 8 times.
✓ Branch 1 taken 1 times.
|
9 | if (!self->match_reported) |
555 | { | ||
556 | 8 | self->match_reported = TRUE; | |
557 | 8 | fpi_device_verify_report (dev, | |
558 | success ? FPI_MATCH_SUCCESS : FPI_MATCH_FAIL, | ||
559 | 8 | g_steal_pointer (&new_scan), | |
560 | NULL); | ||
561 | } | ||
562 | } | ||
563 |
1/2✓ Branch 0 taken 11 times.
✗ Branch 1 not taken.
|
11 | else if (error) |
564 | { | ||
565 | 11 | g_debug ("Virtual device scan failed with error: %s", error->message); | |
566 | } | ||
567 | |||
568 | 20 | fpi_device_report_finger_status_changes (FP_DEVICE (self), | |
569 | FP_FINGER_STATUS_NONE, | ||
570 | FP_FINGER_STATUS_PRESENT); | ||
571 | |||
572 |
4/4✓ Branch 0 taken 11 times.
✓ Branch 1 taken 9 times.
✓ Branch 3 taken 6 times.
✓ Branch 4 taken 5 times.
|
20 | if (error && error->domain == FP_DEVICE_RETRY) |
573 | 6 | fpi_device_verify_report (dev, FPI_MATCH_ERROR, NULL, g_steal_pointer (&error)); | |
574 | |||
575 |
2/2✓ Branch 1 taken 19 times.
✓ Branch 2 taken 1 times.
|
20 | if (should_wait_to_sleep (self, scan_id, error)) |
576 | return; | ||
577 | |||
578 | 19 | self->match_reported = FALSE; | |
579 | 19 | fpi_device_verify_complete (dev, g_steal_pointer (&error)); | |
580 | } | ||
581 | |||
582 | static void | ||
583 | 274 | dev_enroll (FpDevice *dev) | |
584 | { | ||
585 | 274 | g_autoptr(GError) error = NULL; | |
586 | 274 | FpDeviceVirtualDevice *self = FP_DEVICE_VIRTUAL_DEVICE (dev); | |
587 | 274 | FpPrint *print = NULL; | |
588 |
2/4✗ Branch 0 not taken.
✓ Branch 1 taken 175 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 99 times.
|
274 | g_autofree char *id = NULL; |
589 | |||
590 |
2/2✓ Branch 1 taken 182 times.
✓ Branch 2 taken 92 times.
|
274 | if (!start_scan_command (self, &id, &error)) |
591 | return; | ||
592 | |||
593 | 182 | fpi_device_get_enroll_data (dev, &print); | |
594 | |||
595 |
2/2✓ Branch 0 taken 156 times.
✓ Branch 1 taken 26 times.
|
182 | if (id) |
596 | { | ||
597 | 156 | GVariant *data; | |
598 | 156 | gboolean completed; | |
599 | |||
600 |
4/4✓ Branch 0 taken 101 times.
✓ Branch 1 taken 55 times.
✓ Branch 3 taken 1 times.
✓ Branch 4 taken 100 times.
|
156 | if (self->prints_storage && g_hash_table_contains (self->prints_storage, id)) |
601 | { | ||
602 |
1/2✓ Branch 1 taken 1 times.
✗ Branch 2 not taken.
|
1 | if (should_wait_to_sleep (self, id, error)) |
603 | 7 | return; | |
604 | |||
605 | 1 | fpi_device_enroll_complete (dev, NULL, | |
606 | fpi_device_error_new (FP_DEVICE_ERROR_DATA_DUPLICATE)); | ||
607 | 1 | return; | |
608 | } | ||
609 | |||
610 |
2/2✓ Branch 0 taken 29 times.
✓ Branch 1 taken 126 times.
|
155 | if (self->enroll_stages_passed == 0) |
611 | { | ||
612 | 29 | fpi_print_set_type (print, FPI_PRINT_RAW); | |
613 | 29 | data = g_variant_new_string (id); | |
614 | 29 | g_object_set (print, "fpi-data", data, NULL); | |
615 | } | ||
616 | else | ||
617 | { | ||
618 | 126 | gboolean changed; | |
619 | |||
620 | 126 | g_object_get (print, "fpi-data", &data, NULL); | |
621 | 126 | changed = !g_str_equal (id, g_variant_get_string (data, NULL)); | |
622 | 126 | g_variant_unref (data); | |
623 | |||
624 |
2/2✓ Branch 0 taken 6 times.
✓ Branch 1 taken 120 times.
|
126 | if (changed) |
625 | { | ||
626 | 6 | g_set_error (&error, FP_DEVICE_RETRY, FP_DEVICE_RETRY_GENERAL, "ID Mismatch"); | |
627 | 6 | fpi_device_enroll_progress (dev, self->enroll_stages_passed, NULL, | |
628 | 6 | g_steal_pointer (&error)); | |
629 | |||
630 |
1/2✓ Branch 1 taken 6 times.
✗ Branch 2 not taken.
|
6 | if (!should_wait_to_sleep (self, id, error)) |
631 | 6 | self->sleep_timeout_id = g_idle_add (sleep_timeout_cb, self); | |
632 | 6 | return; | |
633 | } | ||
634 | } | ||
635 | |||
636 | 149 | self->enroll_stages_passed++; | |
637 | 149 | completed = self->enroll_stages_passed == fp_device_get_nr_enroll_stages (FP_DEVICE (self)); | |
638 | 149 | fpi_device_report_finger_status_changes (FP_DEVICE (self), | |
639 | completed ? | ||
640 | FP_FINGER_STATUS_NEEDED : | ||
641 | FP_FINGER_STATUS_NONE, | ||
642 | FP_FINGER_STATUS_PRESENT); | ||
643 | |||
644 | 149 | fpi_device_enroll_progress (dev, self->enroll_stages_passed, print, NULL); | |
645 | |||
646 |
2/2✓ Branch 0 taken 29 times.
✓ Branch 1 taken 120 times.
|
149 | if (completed) |
647 | { | ||
648 |
2/2✓ Branch 0 taken 19 times.
✓ Branch 1 taken 10 times.
|
29 | if (self->prints_storage) |
649 | { | ||
650 | 19 | fpi_print_set_device_stored (print, TRUE); | |
651 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 19 times.
|
38 | g_hash_table_add (self->prints_storage, g_strdup (id)); |
652 | } | ||
653 | |||
654 | 29 | fpi_device_enroll_complete (dev, g_object_ref (print), NULL); | |
655 | 29 | self->enroll_stages_passed = 0; | |
656 | } | ||
657 |
2/2✓ Branch 1 taken 114 times.
✓ Branch 2 taken 6 times.
|
120 | else if (!should_wait_to_sleep (self, id, error)) |
658 | { | ||
659 | 114 | self->sleep_timeout_id = g_idle_add (sleep_timeout_cb, self); | |
660 | } | ||
661 | } | ||
662 | else | ||
663 | { | ||
664 | 26 | fpi_device_report_finger_status_changes (FP_DEVICE (self), | |
665 | FP_FINGER_STATUS_NONE, | ||
666 | FP_FINGER_STATUS_PRESENT); | ||
667 | |||
668 |
2/4✓ Branch 0 taken 26 times.
✗ Branch 1 not taken.
✓ Branch 3 taken 26 times.
✗ Branch 4 not taken.
|
26 | if (error && error->domain == FP_DEVICE_RETRY) |
669 | { | ||
670 | 26 | fpi_device_enroll_progress (dev, self->enroll_stages_passed, NULL, g_steal_pointer (&error)); | |
671 | |||
672 |
1/2✓ Branch 1 taken 26 times.
✗ Branch 2 not taken.
|
26 | if (!should_wait_to_sleep (self, id, error)) |
673 | 26 | self->sleep_timeout_id = g_idle_add (sleep_timeout_cb, self); | |
674 | } | ||
675 | else | ||
676 | { | ||
677 | ✗ | if (should_wait_to_sleep (self, id, error)) | |
678 | return; | ||
679 | |||
680 | ✗ | self->enroll_stages_passed = 0; | |
681 | ✗ | fpi_device_enroll_complete (dev, NULL, g_steal_pointer (&error)); | |
682 | } | ||
683 | } | ||
684 | } | ||
685 | |||
686 | static void | ||
687 | 8 | dev_cancel (FpDevice *dev) | |
688 | { | ||
689 | 8 | FpDeviceVirtualDevice *self = FP_DEVICE_VIRTUAL_DEVICE (dev); | |
690 | |||
691 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 8 times.
|
8 | if (self->injected_synthetic_cmd) |
692 | { | ||
693 | ✗ | self->injected_synthetic_cmd = FALSE; | |
694 | ✗ | g_ptr_array_remove_index (self->pending_commands, 0); | |
695 | } | ||
696 | |||
697 |
2/2✓ Branch 0 taken 6 times.
✓ Branch 1 taken 2 times.
|
8 | if (!self->supports_cancellation) |
698 | return; | ||
699 | |||
700 | 6 | g_debug ("Got cancellation!"); | |
701 |
2/2✓ Branch 0 taken 2 times.
✓ Branch 1 taken 4 times.
|
6 | g_clear_handle_id (&self->sleep_timeout_id, g_source_remove); |
702 |
1/2✗ Branch 0 not taken.
✓ Branch 1 taken 6 times.
|
6 | g_clear_handle_id (&self->wait_command_id, g_source_remove); |
703 | |||
704 | 6 | maybe_continue_current_action (self); | |
705 | } | ||
706 | |||
707 | static void | ||
708 | 161 | stop_listener (FpDeviceVirtualDevice *self) | |
709 | { | ||
710 | 161 | g_cancellable_cancel (self->cancellable); | |
711 |
2/2✓ Branch 0 taken 88 times.
✓ Branch 1 taken 73 times.
|
161 | g_clear_object (&self->cancellable); |
712 |
2/2✓ Branch 0 taken 88 times.
✓ Branch 1 taken 73 times.
|
161 | g_clear_object (&self->listener); |
713 | 161 | } | |
714 | |||
715 | static void | ||
716 | 103 | dev_deinit (FpDevice *dev) | |
717 | { | ||
718 | 103 | g_autoptr(GError) error = NULL; | |
719 | 103 | FpDeviceVirtualDevice *self = FP_DEVICE_VIRTUAL_DEVICE (dev); | |
720 | |||
721 | 103 | self->ignore_wait = TRUE; | |
722 |
2/2✓ Branch 1 taken 10 times.
✓ Branch 2 taken 93 times.
|
103 | if (!process_cmds (self, FALSE, NULL, &error)) |
723 | { | ||
724 | 10 | self->ignore_wait = FALSE; | |
725 | 10 | return; | |
726 | } | ||
727 | 93 | self->ignore_wait = FALSE; | |
728 | |||
729 |
2/2✓ Branch 0 taken 2 times.
✓ Branch 1 taken 91 times.
|
93 | if (error) |
730 | { | ||
731 | 2 | fpi_device_close_complete (dev, g_steal_pointer (&error)); | |
732 | 2 | return; | |
733 | } | ||
734 | |||
735 |
2/2✓ Branch 0 taken 83 times.
✓ Branch 1 taken 8 times.
|
91 | if (!self->keep_alive) |
736 | { | ||
737 | 83 | stop_listener (self); | |
738 | 83 | self->supports_cancellation = TRUE; | |
739 | } | ||
740 | |||
741 | 91 | self->enroll_stages_passed = 0; | |
742 | 91 | self->match_reported = FALSE; | |
743 | |||
744 |
1/2✗ Branch 1 not taken.
✓ Branch 2 taken 91 times.
|
91 | fpi_device_close_complete (dev, NULL); |
745 | } | ||
746 | |||
747 | static void | ||
748 | 78 | fpi_device_virtual_device_finalize (GObject *object) | |
749 | { | ||
750 | 78 | FpDeviceVirtualDevice *self = FP_DEVICE_VIRTUAL_DEVICE (object); | |
751 | |||
752 | 78 | G_DEBUG_HERE (); | |
753 | 78 | stop_listener (self); | |
754 |
1/2✓ Branch 0 taken 78 times.
✗ Branch 1 not taken.
|
78 | g_clear_pointer (&self->pending_commands, g_ptr_array_unref); |
755 | 78 | G_OBJECT_CLASS (fpi_device_virtual_device_parent_class)->finalize (object); | |
756 | 78 | } | |
757 | |||
758 | static void | ||
759 | 86 | fpi_device_virtual_device_init (FpDeviceVirtualDevice *self) | |
760 | { | ||
761 | 86 | self->supports_cancellation = TRUE; | |
762 | 86 | self->pending_commands = g_ptr_array_new_with_free_func (g_free); | |
763 | 86 | } | |
764 | |||
765 | static const FpIdEntry driver_ids[] = { | ||
766 | { .virtual_envvar = "FP_VIRTUAL_DEVICE", }, | ||
767 | { .virtual_envvar = NULL } | ||
768 | }; | ||
769 | |||
770 | static void | ||
771 | 120 | fpi_device_virtual_device_class_init (FpDeviceVirtualDeviceClass *klass) | |
772 | { | ||
773 | 120 | FpDeviceClass *dev_class = FP_DEVICE_CLASS (klass); | |
774 | 120 | GObjectClass *object_class = G_OBJECT_CLASS (klass); | |
775 | |||
776 | 120 | object_class->finalize = fpi_device_virtual_device_finalize; | |
777 | |||
778 | 120 | dev_class->id = FP_COMPONENT; | |
779 | 120 | dev_class->full_name = "Virtual device for debugging"; | |
780 | 120 | dev_class->type = FP_DEVICE_TYPE_VIRTUAL; | |
781 | 120 | dev_class->id_table = driver_ids; | |
782 | 120 | dev_class->nr_enroll_stages = 5; | |
783 | |||
784 | 120 | dev_class->open = dev_init; | |
785 | 120 | dev_class->close = dev_deinit; | |
786 | 120 | dev_class->verify = dev_verify; | |
787 | 120 | dev_class->enroll = dev_enroll; | |
788 | 120 | dev_class->cancel = dev_cancel; | |
789 | |||
790 | 120 | fpi_device_class_auto_initialize_features (dev_class); | |
791 | 120 | } | |
792 |