diff options
| author | Benjamin Herr <ben@0x539.de> | 2014-09-30 14:43:03 +0200 |
|---|---|---|
| committer | Pekka Paalanen <pekka.paalanen@collabora.co.uk> | 2014-11-04 11:26:22 +0200 |
| commit | 391820b0d6d9fcd99e12cd32623a476da64c89ce (patch) | |
| tree | ba9b4ea911e7eb3c9dc41e6518b2d4cda419ec04 | |
| parent | doc: Translate doxygen <sp/> tags to spaces (diff) | |
| download | wayland-391820b0d6d9fcd99e12cd32623a476da64c89ce.tar wayland-391820b0d6d9fcd99e12cd32623a476da64c89ce.tar.gz wayland-391820b0d6d9fcd99e12cd32623a476da64c89ce.tar.bz2 wayland-391820b0d6d9fcd99e12cd32623a476da64c89ce.tar.lz wayland-391820b0d6d9fcd99e12cd32623a476da64c89ce.tar.xz wayland-391820b0d6d9fcd99e12cd32623a476da64c89ce.tar.zst wayland-391820b0d6d9fcd99e12cd32623a476da64c89ce.zip | |
connection: Leave fd open in wl_connection_destroy
Calling close() on the same file descriptor that a previous call to
close() already closed is wrong, and racy if another thread received
that same file descriptor as a eg. new socket or actual file.
There are two situations where wl_connection_destroy() would close its
file descriptor and then another function up in the call chain would
close the same file descriptor:
* When wl_client_create() fails after calling wl_connection_create(),
it will call wl_connection_destroy() before returning. However, its
caller will always close the file descriptor if wl_client_create()
fails.
* wl_display_disconnect() unconditionally closes the display file
descriptor and also calls wl_connection_destroy().
So these two seem to expect wl_connection_destroy() to leave the file
descriptor open. The other caller of wl_connection_destroy(),
wl_client_destroy(), does however expect wl_connection_destroy() to
close its file descriptor, alas.
This patch changes wl_connection_destroy() to indulge this majority of
two callers by simply not closing the file descriptor. For the benefit
of wl_client_destroy(), wl_connection_destroy() then returns the
unclosed file descriptor so that wl_client_destroy() can close it
itself.
Since wl_connection_destroy() is a private function called from few
places, changing its semantics seemed like the more expedient way to
address the double-close() problem than shuffling around the logic in
wl_client_create() to somehow enable it to always avoid calling
wl_connection_destroy().
Signed-off-by: Benjamin Herr <ben@0x539.de>
Reviewed-by: Marek Chalupa <mchqwerty@gmail.com>
Reviewed-by: Pekka Paalanen <pekka.paalanen@collabora.co.uk>
| -rw-r--r-- | src/connection.c | 7 | ||||
| -rw-r--r-- | src/wayland-private.h | 2 | ||||
| -rw-r--r-- | src/wayland-server.c | 2 | ||||
| -rw-r--r-- | tests/connection-test.c | 8 |
4 files changed, 13 insertions, 6 deletions
diff --git a/src/connection.c b/src/connection.c index f292853..c5daca6 100644 --- a/src/connection.c +++ b/src/connection.c @@ -189,13 +189,16 @@ close_fds(struct wl_buffer *buffer, int max) buffer->tail += size; } -void +int wl_connection_destroy(struct wl_connection *connection) { + int fd = connection->fd; + close_fds(&connection->fds_out, -1); close_fds(&connection->fds_in, -1); - close(connection->fd); free(connection); + + return fd; } void diff --git a/src/wayland-private.h b/src/wayland-private.h index 67e8783..db76081 100644 --- a/src/wayland-private.h +++ b/src/wayland-private.h @@ -86,7 +86,7 @@ int wl_interface_equal(const struct wl_interface *iface1, const struct wl_interface *iface2); struct wl_connection *wl_connection_create(int fd); -void wl_connection_destroy(struct wl_connection *connection); +int wl_connection_destroy(struct wl_connection *connection); void wl_connection_copy(struct wl_connection *connection, void *data, size_t size); void wl_connection_consume(struct wl_connection *connection, size_t size); diff --git a/src/wayland-server.c b/src/wayland-server.c index 674aeca..7caeb30 100644 --- a/src/wayland-server.c +++ b/src/wayland-server.c @@ -670,7 +670,7 @@ wl_client_destroy(struct wl_client *client) wl_map_for_each(&client->objects, destroy_resource, &serial); wl_map_release(&client->objects); wl_event_source_remove(client->source); - wl_connection_destroy(client->connection); + close(wl_connection_destroy(client->connection)); wl_list_remove(&client->link); free(client); } diff --git a/tests/connection-test.c b/tests/connection-test.c index 659bf68..4497128 100644 --- a/tests/connection-test.c +++ b/tests/connection-test.c @@ -57,6 +57,7 @@ TEST(connection_create) connection = setup(s); wl_connection_destroy(connection); + close(s[0]); close(s[1]); } @@ -74,6 +75,7 @@ TEST(connection_write) assert(memcmp(message, buffer, sizeof message) == 0); wl_connection_destroy(connection); + close(s[0]); close(s[1]); } @@ -92,6 +94,7 @@ TEST(connection_data) wl_connection_consume(connection, sizeof message); wl_connection_destroy(connection); + close(s[0]); close(s[1]); } @@ -117,6 +120,7 @@ TEST(connection_queue) assert(memcmp(message, buffer + sizeof message, sizeof message) == 0); wl_connection_destroy(connection); + close(s[0]); close(s[1]); } @@ -147,8 +151,8 @@ setup_marshal_data(struct marshal_data *data) static void release_marshal_data(struct marshal_data *data) { - wl_connection_destroy(data->read_connection); - wl_connection_destroy(data->write_connection); + close(wl_connection_destroy(data->read_connection)); + close(wl_connection_destroy(data->write_connection)); } static void |
