| Line | Branch | Exec | Source |
|---|---|---|---|
| 1 | /* | ||
| 2 | * Socket utilities for "simple" device debugging | ||
| 3 | * | ||
| 4 | * Copyright (C) 2019 Benjamin Berg <bberg@redhat.com> | ||
| 5 | * Copyright (C) 2020 Marco Trevisan <marco.trevisan@canonical.com> | ||
| 6 | * | ||
| 7 | * This library is free software; you can redistribute it and/or | ||
| 8 | * modify it under the terms of the GNU Lesser General Public | ||
| 9 | * License as published by the Free Software Foundation; either | ||
| 10 | * version 2.1 of the License, or (at your option) any later version. | ||
| 11 | * | ||
| 12 | * This library is distributed in the hope that it will be useful, | ||
| 13 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
| 14 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | ||
| 15 | * Lesser General Public License for more details. | ||
| 16 | * | ||
| 17 | * You should have received a copy of the GNU Lesser General Public | ||
| 18 | * License along with this library; if not, write to the Free Software | ||
| 19 | * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA | ||
| 20 | */ | ||
| 21 | |||
| 22 | #define FP_COMPONENT "virtual_device_connection" | ||
| 23 | |||
| 24 | #include "fpi-log.h" | ||
| 25 | |||
| 26 | #include <glib/gstdio.h> | ||
| 27 | #include <gio/gunixsocketaddress.h> | ||
| 28 | |||
| 29 | #include "virtual-device-private.h" | ||
| 30 | |||
| 31 | struct _FpiDeviceVirtualListener | ||
| 32 | { | ||
| 33 | GSocketListener parent_instance; | ||
| 34 | |||
| 35 | GSocketConnection *connection; | ||
| 36 | GCancellable *cancellable; | ||
| 37 | guint cancellable_id; | ||
| 38 | |||
| 39 | FpiDeviceVirtualListenerConnectionCb ready_cb; | ||
| 40 | gpointer ready_cb_data; | ||
| 41 | |||
| 42 | gint socket_fd; | ||
| 43 | gint client_fd; | ||
| 44 | }; | ||
| 45 | |||
| 46 |
3/4✓ Branch 0 (2→3) taken 93 times.
✓ Branch 1 (2→7) taken 1389 times.
✓ Branch 2 (4→5) taken 93 times.
✗ Branch 3 (4→7) not taken.
|
1575 | G_DEFINE_TYPE (FpiDeviceVirtualListener, fpi_device_virtual_listener, G_TYPE_SOCKET_LISTENER) |
| 47 | |||
| 48 | static void start_listen (FpiDeviceVirtualListener *self); | ||
| 49 | |||
| 50 | FpiDeviceVirtualListener * | ||
| 51 | 114 | fpi_device_virtual_listener_new (void) | |
| 52 | { | ||
| 53 | 114 | return g_object_new (fpi_device_virtual_listener_get_type (), NULL); | |
| 54 | } | ||
| 55 | |||
| 56 | static void | ||
| 57 | 39 | fpi_device_virtual_listener_dispose (GObject *object) | |
| 58 | { | ||
| 59 | 39 | FpiDeviceVirtualListener *self = FPI_DEVICE_VIRTUAL_LISTENER (object); | |
| 60 | |||
| 61 |
1/2✓ Branch 0 (2→3) taken 39 times.
✗ Branch 1 (2→5) not taken.
|
39 | if (self->cancellable_id) |
| 62 | { | ||
| 63 | 39 | g_cancellable_disconnect (self->cancellable, self->cancellable_id); | |
| 64 | 39 | self->cancellable_id = 0; | |
| 65 | } | ||
| 66 | |||
| 67 | 39 | g_cancellable_cancel (self->cancellable); | |
| 68 |
1/2✗ Branch 0 (6→7) not taken.
✓ Branch 1 (6→8) taken 39 times.
|
39 | g_clear_object (&self->cancellable); |
| 69 |
1/2✗ Branch 0 (8→9) not taken.
✓ Branch 1 (8→10) taken 39 times.
|
39 | g_clear_object (&self->connection); |
| 70 | |||
| 71 | 39 | self->ready_cb = NULL; | |
| 72 | |||
| 73 | 39 | G_OBJECT_CLASS (fpi_device_virtual_listener_parent_class)->dispose (object); | |
| 74 | 39 | } | |
| 75 | |||
| 76 | static void | ||
| 77 | 93 | fpi_device_virtual_listener_class_init (FpiDeviceVirtualListenerClass *klass) | |
| 78 | { | ||
| 79 | 93 | GObjectClass *object_class = G_OBJECT_CLASS (klass); | |
| 80 | |||
| 81 | 93 | object_class->dispose = fpi_device_virtual_listener_dispose; | |
| 82 | } | ||
| 83 | |||
| 84 | static void | ||
| 85 | 114 | fpi_device_virtual_listener_init (FpiDeviceVirtualListener *self) | |
| 86 | { | ||
| 87 | 114 | } | |
| 88 | |||
| 89 | static void | ||
| 90 | 510 | new_connection_cb (GObject *source_object, GAsyncResult *res, gpointer user_data) | |
| 91 | { | ||
| 92 | 510 | g_autoptr(GError) error = NULL; | |
| 93 | 510 | FpiDeviceVirtualListener *self = user_data; | |
| 94 | 510 | GSocketConnection *connection; | |
| 95 | |||
| 96 | 510 | connection = g_socket_listener_accept_finish (G_SOCKET_LISTENER (source_object), | |
| 97 | res, | ||
| 98 | NULL, | ||
| 99 | &error); | ||
| 100 |
2/2✓ Branch 0 (3→4) taken 39 times.
✓ Branch 1 (3→9) taken 471 times.
|
510 | if (!connection) |
| 101 | { | ||
| 102 |
1/2✗ Branch 0 (6→7) not taken.
✓ Branch 1 (6→20) taken 39 times.
|
39 | if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED)) |
| 103 | return; | ||
| 104 | |||
| 105 | ✗ | g_warning ("Error accepting a new connection: %s", error->message); | |
| 106 | ✗ | start_listen (self); | |
| 107 | ✗ | return; | |
| 108 | } | ||
| 109 | |||
| 110 | /* Always allow further connections. | ||
| 111 | * If we get a new one, we generally just close the old connection. */ | ||
| 112 | 471 | start_listen (self); | |
| 113 |
1/2✗ Branch 0 (10→11) not taken.
✓ Branch 1 (10→14) taken 471 times.
|
471 | if (self->connection) |
| 114 | { | ||
| 115 | ✗ | g_io_stream_close (G_IO_STREAM (self->connection), NULL, NULL); | |
| 116 | ✗ | g_clear_object (&self->connection); | |
| 117 | } | ||
| 118 | |||
| 119 | 471 | self->connection = connection; | |
| 120 | 471 | fp_dbg ("Got a new connection!"); | |
| 121 | |||
| 122 |
1/2✗ Branch 0 (16→17) not taken.
✓ Branch 1 (16→18) taken 471 times.
|
471 | self->ready_cb (self, self->ready_cb_data); |
| 123 | } | ||
| 124 | |||
| 125 | static void | ||
| 126 | 585 | start_listen (FpiDeviceVirtualListener *self) | |
| 127 | { | ||
| 128 | 585 | g_socket_listener_accept_async (G_SOCKET_LISTENER (self), | |
| 129 | self->cancellable, | ||
| 130 | new_connection_cb, | ||
| 131 | self); | ||
| 132 | 585 | } | |
| 133 | |||
| 134 | static void | ||
| 135 | 114 | on_cancelled (GCancellable *cancellable, | |
| 136 | FpiDeviceVirtualListener *self) | ||
| 137 | { | ||
| 138 | 114 | fpi_device_virtual_listener_connection_close (self); | |
| 139 | 114 | g_socket_listener_close (G_SOCKET_LISTENER (self)); | |
| 140 |
1/2✓ Branch 0 (4→5) taken 114 times.
✗ Branch 1 (4→6) not taken.
|
114 | g_clear_object (&self->cancellable); |
| 141 | 114 | self->ready_cb = NULL; | |
| 142 | 114 | } | |
| 143 | |||
| 144 | gboolean | ||
| 145 | 114 | fpi_device_virtual_listener_start (FpiDeviceVirtualListener *self, | |
| 146 | const char *address, | ||
| 147 | GCancellable *cancellable, | ||
| 148 | FpiDeviceVirtualListenerConnectionCb cb, | ||
| 149 | gpointer user_data, | ||
| 150 | GError **error) | ||
| 151 | { | ||
| 152 | 228 | g_autoptr(GSocketAddress) addr = NULL; | |
| 153 | 114 | G_DEBUG_HERE (); | |
| 154 | |||
| 155 |
1/2✗ Branch 0 (5→6) not taken.
✓ Branch 1 (5→7) taken 114 times.
|
114 | g_return_val_if_fail (FPI_IS_DEVICE_VIRTUAL_LISTENER (self), FALSE); |
| 156 |
1/2✓ Branch 0 (7→8) taken 114 times.
✗ Branch 1 (7→9) not taken.
|
114 | g_return_val_if_fail (cb != NULL, FALSE); |
| 157 |
1/2✓ Branch 0 (8→10) taken 114 times.
✗ Branch 1 (8→15) not taken.
|
114 | g_return_val_if_fail (self->ready_cb == NULL, FALSE); |
| 158 | |||
| 159 | 114 | self->client_fd = -1; | |
| 160 | |||
| 161 | 114 | g_socket_listener_set_backlog (G_SOCKET_LISTENER (self), 1); | |
| 162 | |||
| 163 | /* Remove any left over socket. */ | ||
| 164 | 114 | g_unlink (address); | |
| 165 | |||
| 166 | 114 | addr = g_unix_socket_address_new (address); | |
| 167 | |||
| 168 |
1/2✗ Branch 0 (14→17) not taken.
✓ Branch 1 (14→19) taken 114 times.
|
114 | if (!g_socket_listener_add_address (G_SOCKET_LISTENER (self), |
| 169 | addr, | ||
| 170 | G_SOCKET_TYPE_STREAM, | ||
| 171 | G_SOCKET_PROTOCOL_DEFAULT, | ||
| 172 | NULL, | ||
| 173 | NULL, | ||
| 174 | error)) | ||
| 175 | { | ||
| 176 | ✗ | g_warning ("Could not listen on unix socket: %s", (*error)->message); | |
| 177 | ✗ | return FALSE; | |
| 178 | } | ||
| 179 | |||
| 180 | 114 | self->ready_cb = cb; | |
| 181 | 114 | self->ready_cb_data = user_data; | |
| 182 |
1/2✓ Branch 0 (19→20) taken 114 times.
✗ Branch 1 (19→21) not taken.
|
114 | self->cancellable = cancellable ? g_object_ref (cancellable) : NULL; |
| 183 | |||
| 184 |
1/2✓ Branch 0 (21→22) taken 114 times.
✗ Branch 1 (21→24) not taken.
|
114 | if (self->cancellable) |
| 185 | 114 | self->cancellable_id = g_cancellable_connect (self->cancellable, | |
| 186 | G_CALLBACK (on_cancelled), self, NULL); | ||
| 187 | |||
| 188 | 114 | start_listen (self); | |
| 189 | |||
| 190 | 114 | return TRUE; | |
| 191 | } | ||
| 192 | |||
| 193 | gboolean | ||
| 194 | 581 | fpi_device_virtual_listener_connection_close (FpiDeviceVirtualListener *self) | |
| 195 | { | ||
| 196 |
1/2✗ Branch 0 (3→4) not taken.
✓ Branch 1 (3→6) taken 581 times.
|
581 | g_return_val_if_fail (FPI_IS_DEVICE_VIRTUAL_LISTENER (self), FALSE); |
| 197 | |||
| 198 |
2/2✓ Branch 0 (6→5) taken 110 times.
✓ Branch 1 (6→7) taken 471 times.
|
581 | if (!self->connection) |
| 199 | return FALSE; | ||
| 200 | |||
| 201 | 471 | g_io_stream_close (G_IO_STREAM (self->connection), NULL, NULL); | |
| 202 |
1/2✓ Branch 0 (8→9) taken 471 times.
✗ Branch 1 (8→10) not taken.
|
471 | g_clear_object (&self->connection); |
| 203 | |||
| 204 | return TRUE; | ||
| 205 | } | ||
| 206 | |||
| 207 | static void | ||
| 208 | 575 | on_stream_read_cb (GObject *source_object, | |
| 209 | GAsyncResult *res, | ||
| 210 | gpointer user_data) | ||
| 211 | { | ||
| 212 | 575 | g_autoptr(GError) error = NULL; | |
| 213 |
3/4✗ Branch 0 (38→39) not taken.
✓ Branch 1 (38→40) taken 557 times.
✓ Branch 2 (44→45) taken 4 times.
✓ Branch 3 (44→46) taken 14 times.
|
575 | g_autoptr(GTask) task = user_data; |
| 214 | 575 | FpiDeviceVirtualListener *self = g_task_get_source_object (task); | |
| 215 | 575 | gboolean all; | |
| 216 | 575 | gboolean success; | |
| 217 | 575 | gsize bytes; | |
| 218 | |||
| 219 | 575 | all = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (task), "all")); | |
| 220 | |||
| 221 |
2/2✓ Branch 0 (4→5) taken 108 times.
✓ Branch 1 (4→6) taken 467 times.
|
575 | if (all) |
| 222 | { | ||
| 223 | 108 | success = g_input_stream_read_all_finish (G_INPUT_STREAM (source_object), res, &bytes, &error); | |
| 224 | } | ||
| 225 | else | ||
| 226 | { | ||
| 227 | 467 | gssize sbytes; | |
| 228 | |||
| 229 | 467 | sbytes = g_input_stream_read_finish (G_INPUT_STREAM (source_object), res, &error); | |
| 230 | 467 | bytes = sbytes; | |
| 231 | 467 | success = (sbytes >= 0); | |
| 232 | } | ||
| 233 | |||
| 234 |
2/2✓ Branch 0 (9→10) taken 571 times.
✓ Branch 1 (9→42) taken 4 times.
|
575 | if (g_task_return_error_if_cancelled (task)) |
| 235 | return; | ||
| 236 | |||
| 237 | /* If we are cancelled, just return immediately. */ | ||
| 238 |
1/2✗ Branch 0 (12→13) not taken.
✓ Branch 1 (12→14) taken 571 times.
|
571 | if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CLOSED)) |
| 239 | { | ||
| 240 | ✗ | g_task_return_int (task, 0); | |
| 241 | ✗ | return; | |
| 242 | } | ||
| 243 | |||
| 244 | /* If this error is for an old connection (that should be closed already), | ||
| 245 | * then just give up immediately with a CLOSED error. | ||
| 246 | */ | ||
| 247 |
2/4✓ Branch 0 (14→15) taken 571 times.
✗ Branch 1 (14→20) not taken.
✗ Branch 2 (16→17) not taken.
✓ Branch 3 (16→20) taken 571 times.
|
1142 | if (self->connection && |
| 248 | 571 | g_io_stream_get_input_stream (G_IO_STREAM (self->connection)) != G_INPUT_STREAM (source_object)) | |
| 249 | { | ||
| 250 | ✗ | g_task_return_new_error (task, | |
| 251 | G_IO_ERROR, | ||
| 252 | G_IO_ERROR_CLOSED, | ||
| 253 | "Error on old connection, ignoring."); | ||
| 254 | ✗ | return; | |
| 255 | } | ||
| 256 | |||
| 257 |
3/4✓ Branch 0 (20→21) taken 557 times.
✓ Branch 1 (20→22) taken 14 times.
✗ Branch 2 (21→22) not taken.
✓ Branch 3 (21→35) taken 557 times.
|
571 | if (!success || bytes == 0) |
| 258 | { | ||
| 259 | /* We accept it if someone tries to read twice and just return that error. */ | ||
| 260 |
1/2✗ Branch 0 (24→25) not taken.
✓ Branch 1 (24→29) taken 14 times.
|
14 | if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_PENDING)) |
| 261 | { | ||
| 262 | ✗ | if (self->connection) | |
| 263 | { | ||
| 264 | ✗ | g_io_stream_close (G_IO_STREAM (self->connection), NULL, NULL); | |
| 265 | ✗ | g_clear_object (&self->connection); | |
| 266 | } | ||
| 267 | } | ||
| 268 | |||
| 269 |
1/2✓ Branch 0 (29→30) taken 14 times.
✗ Branch 1 (29→32) not taken.
|
14 | if (error) |
| 270 | { | ||
| 271 | 14 | g_task_return_error (task, g_steal_pointer (&error)); | |
| 272 | 14 | return; | |
| 273 | } | ||
| 274 | else | ||
| 275 | { | ||
| 276 | ✗ | g_task_return_new_error (task, G_IO_ERROR, G_IO_ERROR_INVALID_DATA, | |
| 277 | "Got empty data"); | ||
| 278 | ✗ | return; | |
| 279 | } | ||
| 280 | } | ||
| 281 | |||
| 282 |
1/2✓ Branch 0 (36→37) taken 557 times.
✗ Branch 1 (36→38) not taken.
|
557 | g_task_return_int (task, bytes); |
| 283 | } | ||
| 284 | |||
| 285 | void | ||
| 286 | 580 | fpi_device_virtual_listener_read (FpiDeviceVirtualListener *self, | |
| 287 | gboolean all, | ||
| 288 | void *buffer, | ||
| 289 | gsize count, | ||
| 290 | GAsyncReadyCallback callback, | ||
| 291 | gpointer user_data) | ||
| 292 | { | ||
| 293 | 1160 | g_autoptr(GTask) task = NULL; | |
| 294 | 580 | GInputStream *stream; | |
| 295 | |||
| 296 |
1/2✗ Branch 0 (3→4) not taken.
✓ Branch 1 (3→5) taken 580 times.
|
580 | g_return_if_fail (FPI_IS_DEVICE_VIRTUAL_LISTENER (self)); |
| 297 | |||
| 298 | 580 | task = g_task_new (self, self->cancellable, callback, user_data); | |
| 299 | 580 | g_object_set_data (G_OBJECT (task), "all", GINT_TO_POINTER (all)); | |
| 300 | |||
| 301 |
3/4✓ Branch 0 (7→8) taken 575 times.
✓ Branch 1 (7→10) taken 5 times.
✗ Branch 2 (9→10) not taken.
✓ Branch 3 (9→13) taken 575 times.
|
580 | if (!self->connection || g_io_stream_is_closed (G_IO_STREAM (self->connection))) |
| 302 | { | ||
| 303 | 5 | g_task_return_new_error (task, G_IO_ERROR, G_IO_ERROR_CONNECTION_CLOSED, | |
| 304 | "Listener not connected to any stream"); | ||
| 305 |
1/2✓ Branch 0 (12→17) taken 5 times.
✗ Branch 1 (12→18) not taken.
|
5 | return; |
| 306 | } | ||
| 307 | |||
| 308 | 575 | stream = g_io_stream_get_input_stream (G_IO_STREAM (self->connection)); | |
| 309 |
2/2✓ Branch 0 (14→15) taken 108 times.
✓ Branch 1 (14→16) taken 467 times.
|
575 | if (all) |
| 310 | { | ||
| 311 | 108 | g_input_stream_read_all_async (stream, buffer, count, | |
| 312 | G_PRIORITY_DEFAULT, | ||
| 313 | self->cancellable, | ||
| 314 | on_stream_read_cb, | ||
| 315 | g_steal_pointer (&task)); | ||
| 316 | } | ||
| 317 | else | ||
| 318 | { | ||
| 319 | 467 | g_input_stream_read_async (stream, buffer, count, | |
| 320 | G_PRIORITY_DEFAULT, | ||
| 321 | self->cancellable, | ||
| 322 | on_stream_read_cb, | ||
| 323 | g_steal_pointer (&task)); | ||
| 324 | } | ||
| 325 | } | ||
| 326 | |||
| 327 | gsize | ||
| 328 | 580 | fpi_device_virtual_listener_read_finish (FpiDeviceVirtualListener *self, | |
| 329 | GAsyncResult *result, | ||
| 330 | GError **error) | ||
| 331 | { | ||
| 332 |
1/2✗ Branch 0 (3→4) not taken.
✓ Branch 1 (3→5) taken 580 times.
|
580 | g_return_val_if_fail (g_task_is_valid (result, self), 0); |
| 333 | |||
| 334 | 580 | return g_task_propagate_int (G_TASK (result), error); | |
| 335 | } | ||
| 336 | |||
| 337 | gboolean | ||
| 338 | ✗ | fpi_device_virtual_listener_write_sync (FpiDeviceVirtualListener *self, | |
| 339 | const char *buffer, | ||
| 340 | gsize count, | ||
| 341 | GError **error) | ||
| 342 | { | ||
| 343 | ✗ | if (!self->connection || g_io_stream_is_closed (G_IO_STREAM (self->connection))) | |
| 344 | { | ||
| 345 | ✗ | g_set_error (error, G_IO_ERROR, G_IO_ERROR_CONNECTION_CLOSED, | |
| 346 | "Listener not connected to any stream"); | ||
| 347 | ✗ | return FALSE; | |
| 348 | } | ||
| 349 | |||
| 350 | ✗ | return g_output_stream_write_all (g_io_stream_get_output_stream (G_IO_STREAM (self->connection)), | |
| 351 | buffer, | ||
| 352 | count, | ||
| 353 | NULL, | ||
| 354 | self->cancellable, | ||
| 355 | error); | ||
| 356 | } | ||
| 357 |