Compare commits

...

45 Commits

Author SHA1 Message Date
Slava Monich
f24145f407 Version 1.0.28 2019-01-29 02:50:13 +02:00
Slava Monich
11c173b2e4 Merge pull request #33 from mer-hybris/jb44588
Set type for local nulls to BINDER_TYPE_WEAK_BINDER
2019-01-29 02:48:09 +02:00
Andrew Branson
aff816d10c [binder] Set type for local nulls to BINDER_TYPE_WEAK_BINDER. JB#44588
Else they are not equivalent to null in Java.
2019-01-29 01:09:52 +01:00
Slava Monich
e1acaa3bdb Version 1.0.27 2019-01-24 18:56:58 +02:00
Slava Monich
c63743ac51 Merge pull request #32 from monich/oneway
Fix outgoing oneway transactions
2019-01-24 18:53:51 +02:00
Slava Monich
6832d9bf46 [unit] Added unit test for async oneway transactions 2019-01-24 18:51:26 +02:00
Slava Monich
4c3ccbc06f [gbinder] Fixed outgoing oneway transactions. JB#42956
Oneway transactions didn't have the appropriate flag set.
Also, a reply was unnecessarily allocated for those and
then just deallocated, which was a total waste.
2019-01-24 18:36:20 +02:00
Slava Monich
f82596c372 Version 1.0.26 2019-01-23 17:45:25 +02:00
Slava Monich
4921a6ab8d Merge pull request #28 from monich/interface_tx
Implement PING and INTERFACE transactions
2019-01-23 17:39:07 +02:00
Slava Monich
ef9c242a59 Merge pull request #30 from monich/servicename
Add GBinderServiceName object
2019-01-23 17:34:13 +02:00
Slava Monich
a83c9937a5 Merge pull request #29 from monich/utf16
Add gbinder_reader_read_string16_utf16()
2019-01-23 17:25:40 +02:00
Slava Monich
cfa3ad4d9e [unit] Added unit test for GBinderServiceName 2019-01-23 01:35:56 +02:00
Slava Monich
ffc9638ebb [gbinder] Added GBinderServiceName object. JB#42956
It keeps GBinderLocalObject registered, waiting for servicemanager
to appear and re-registering the object after servicemanager restarts.
2019-01-23 01:34:54 +02:00
Slava Monich
000534654d [gbinder] Added gbinder_reader_read_string16_utf16(). JB#42956
Returns const pointer to UTF-16 string in binder's buffer. The
memory pointed to by this string gets automatically deallocated
by libgbinder when transaction completes.

Also added const to gbinder_reader_read_nullable_string16_utf16()
output argument. It may cause compilation warnings, but that makes
it obvious that caller doesn't deallocate the returned pointer, in
my opinion it's worth the trouble.
2019-01-22 18:39:51 +02:00
Slava Monich
68e9358d02 [gbinder] Added tests for PING and INTERFACE transactions 2019-01-22 17:36:02 +02:00
Slava Monich
37e3859f8f [gbinder] Implement PING and INTERFACE transactions. JB#42956 2019-01-22 17:36:02 +02:00
Slava Monich
e79b940b0d Version 1.0.25 2019-01-22 16:05:32 +02:00
Slava Monich
d8dfe3f289 Merge pull request #27 from monich/sm_presence
ServiceManager presence API
2019-01-22 16:00:35 +02:00
Slava Monich
394c286ee5 [gbinder] Re-arm watches after restart of servicemanager. JB#42956 2019-01-20 18:14:15 +02:00
Slava Monich
307bd6942c [gbinder] Added servicemanager presence API. JB#42956
This function allows to wait for servicemanager synchronously,
blocking the event loop in the process (not recommended):

  gbinder_servicemanager_wait()

These two allow to follow the presence state of servicemanager
without blocking the event loop:

  gbinder_servicemanager_is_present()
  gbinder_servicemanager_add_presence_handler()

Note that services need to re-add their names to servicemanager
after it has restarted. This can be done by the presence handler.
2019-01-20 14:12:25 +02:00
Slava Monich
148b53e862 Version 1.0.24 2019-01-18 21:38:03 +02:00
Slava Monich
5c8cb0a013 Merge pull request #26 from mlehtima/jb44476
Revert "Make sure NULL objects are passed as BINDER_TYPE_WEAK_HANDLE"
2019-01-18 21:34:03 +02:00
Matti Lehtimäki
4d644e0584 [local_object] Revert "Make sure NULL objects are passed as BINDER_TYPE_WEAK_HANDLE". Fixes JB#44476
This reverts commit 3b299d3345 which broke handling of NULL objects in certain situations.
2019-01-18 21:21:06 +02:00
Slava Monich
dce9c8b3d1 Version 1.0.23 2019-01-15 15:17:47 +02:00
Slava Monich
b117ee6404 Merge pull request #25 from monich/buffer_obj
Add gbinder_reader_read_hidl_string_c()
2019-01-15 15:14:10 +02:00
Slava Monich
c3f783bf7e [gbinder] Added gbinder_reader_read_hidl_string_c(). JB#42956
It allows to fetch a pointer to the hidl string contents (which must
be NULL terminated) without duplicating the string and allocating any
memory in the process.

Added gbinder_reader_skip_hidl_string() macro.

Reduced number of memory allocations performed by functions parsing
buffer objects. Many of them don't allocate any memory at all now.
2019-01-15 14:44:25 +02:00
Slava Monich
4c65a6eded [licence] 2018 -> 2019 2019-01-15 01:48:50 +02:00
Slava Monich
c382cab922 [gbinder] Housekeeping
Fixed compilation warning:

gbinder_writer.c:223:9: warning: format ‘%d’ expects argument of type ‘int’,
but argument 4 has type ‘gsize’ [-Wformat=]

and a new other things.
2019-01-11 02:36:25 +03:00
Slava Monich
199fd4ed61 Version 1.0.22 2019-01-10 14:14:35 +03:00
Slava Monich
f86d62fbf8 Merge pull request #24 from monich/client_iface
Add gbinder_client_interface() function
2019-01-10 14:06:01 +03:00
Slava Monich
3ea82dc384 [gbinder] Added gbinder_client_interface(). JB#42956 2019-01-10 12:55:57 +03:00
Slava Monich
03ae5834ee [gbinder] Made warning message more informative. JB#42956 2019-01-10 12:46:23 +03:00
Slava Monich
e39b5c20ee [gbinder] Housekeeping 2019-01-10 12:45:23 +03:00
Slava Monich
d6e131eb6e Version 1.0.21 2018-12-18 14:11:01 +02:00
Slava Monich
1c36b5f142 Acknowledge Andrew's contribution 2018-12-18 14:09:49 +02:00
Slava Monich
827bd0b59f Merge pull request #23 from mer-hybris/overwrite
Add gbinder_writer_bytes_written and gbinder_writer_overwrite_int32
2018-12-18 14:02:07 +02:00
Andrew Branson
2dab057652 [gbinder] Add gbinder_writer_bytes_written and gbinder_writer_overwrite_int32. JB#43524
Bundles require their prefix length value to be set after the rest of the bundle has been written. This needs to remember the buffer position before a dummy value is written, which it then overwrites with the real value afterwards.
2018-12-18 13:00:30 +01:00
Slava Monich
488fbc5b63 Version 1.0.20 2018-12-17 16:07:50 +02:00
Slava Monich
17f511d7a3 Merge pull request #22 from monich/block
Add API to block incoming requests
2018-12-17 16:04:49 +02:00
Slava Monich
63e633c0ec [test] Added option to test asynchronous completion 2018-12-17 15:46:28 +02:00
Slava Monich
f46448c236 [gbinder] Added API to block incoming requests. JB#43529
GBinderRemoteRequest can be marked as blocked by calling
gbinder_remote_request_block() and later completed with
gbinder_remote_request_complete()
2018-12-17 15:46:19 +02:00
Slava Monich
43023be32d [test] Fixed typo 2018-12-17 00:36:21 +02:00
Slava Monich
4811d51c5d Version 1.0.19 2018-12-14 16:40:31 +02:00
Slava Monich
821dabca3d Merge pull request #21 from monich/cleanup
GBinderWriter allocators and cleanup
2018-12-14 16:01:31 +02:00
Slava Monich
587f4ebb50 [gbinder] GBinderWriter allocators and cleanup. JB#42956
Memory allocated by these allocators will be deallocated together
with the rest of the data.
2018-12-13 15:05:44 +02:00
50 changed files with 4261 additions and 602 deletions

View File

@@ -2,3 +2,4 @@ Slava Monich <slava.monich@jolla.com>
Matti Lehtimäki <matti.lehtimaki@gmail.com>
Franz-Josef Haider <franz.haider@jolla.com>
Juho Hämäläinen <juho.hamalainen@jolla.com>
Andrew Branson <andrew.branson@jolla.com>

10
LICENSE
View File

@@ -1,5 +1,5 @@
Copyright (C) 2018 Jolla Ltd.
Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
Copyright (C) 2018-2019 Jolla Ltd.
Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
You may use this file under the terms of BSD license as follows:
@@ -12,9 +12,9 @@ are met:
2. Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
3. Neither the name of Jolla Ltd nor the names of its contributors may
be used to endorse or promote products derived from this software
without specific prior written permission.
3. Neither the names of the copyright holders nor the names of its
contributors may be used to endorse or promote products derived
from this software without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE

View File

@@ -24,7 +24,7 @@ all: debug release pkgconfig
VERSION_MAJOR = 1
VERSION_MINOR = 0
VERSION_RELEASE = 18
VERSION_RELEASE = 28
# Version for pkg-config
PCVERSION = $(VERSION_MAJOR).$(VERSION_MINOR).$(VERSION_RELEASE)
@@ -62,6 +62,7 @@ SRC = \
gbinder_remote_reply.c \
gbinder_remote_request.c \
gbinder_rpc_protocol.c \
gbinder_servicename.c \
gbinder_servicepoll.c \
gbinder_writer.c

66
debian/changelog vendored
View File

@@ -1,3 +1,69 @@
libgbinder (1.0.28) unstable; urgency=low
* Set type for local nulls to BINDER_TYPE_WEAK_BINDER
-- Slava Monich <slava.monich@jolla.com> Tue, 29 Jan 2019 02:49:10 +0200
libgbinder (1.0.27) unstable; urgency=low
* Fixed outgoing oneway transactions
-- Slava Monich <slava.monich@jolla.com> Thu, 24 Jan 2019 18:55:16 +0200
libgbinder (1.0.26) unstable; urgency=low
* Implement PING and INTERFACE transactions
* Add GBinderServiceName API
* Added gbinder_reader_read_string16_utf16()
-- Slava Monich <slava.monich@jolla.com> Wed, 23 Jan 2019 17:43:41 +0200
libgbinder (1.0.25) unstable; urgency=low
* Added ServiceManager presence API
gbinder_servicemanager_wait()
gbinder_servicemanager_is_present()
gbinder_servicemanager_add_presence_handler()
-- Slava Monich <slava.monich@jolla.com> Tue, 22 Jan 2019 16:03:57 +0200
libgbinder (1.0.24) unstable; urgency=low
* Revert "Make sure NULL objects are passed as BINDER_TYPE_WEAK_HANDLE"
-- Slava Monich <slava.monich@jolla.com> Fri, 18 Jan 2019 21:36:32 +0200
libgbinder (1.0.23) unstable; urgency=low
* Added gbinder_reader_read_hidl_string_c()
-- Slava Monich <slava.monich@jolla.com> Tue, 15 Jan 2019 15:16:41 +0200
libgbinder (1.0.22) unstable; urgency=low
* Added gbinder_client_interface()
-- Slava Monich <slava.monich@jolla.com> Thu, 10 Jan 2019 14:09:44 +0300
libgbinder (1.0.21) unstable; urgency=low
* Added API to overwrite prefix length
-- Slava Monich <slava.monich@jolla.com> Tue, 18 Dec 2018 14:05:14 +0200
libgbinder (1.0.20) unstable; urgency=low
* Added API to block incoming requests
-- Slava Monich <slava.monich@jolla.com> Mon, 17 Dec 2018 16:06:43 +0200
libgbinder (1.0.19) unstable; urgency=low
* Added GBinderWriter memory allocation and cleanup API
-- Slava Monich <slava.monich@jolla.com> Fri, 14 Dec 2018 16:27:51 +0200
libgbinder (1.0.18) unstable; urgency=low
* Implemented support for file descritors

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -13,9 +13,9 @@
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of Jolla Ltd nor the names of its contributors may
* be used to endorse or promote products derived from this software
* without specific prior written permission.
* 3. Neither the names of the copyright holders nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
@@ -44,6 +44,7 @@
#include "gbinder_remote_object.h"
#include "gbinder_remote_reply.h"
#include "gbinder_remote_request.h"
#include "gbinder_servicename.h"
#include "gbinder_servicemanager.h"
#include "gbinder_writer.h"

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -13,9 +13,9 @@
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of Jolla Ltd nor the names of its contributors may
* be used to endorse or promote products derived from this software
* without specific prior written permission.
* 3. Neither the names of the copyright holders nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
@@ -58,6 +58,10 @@ void
gbinder_client_unref(
GBinderClient* client);
const char*
gbinder_client_interface(
GBinderClient* client); /* since 1.0.22 */
GBinderLocalRequest*
gbinder_client_new_request(
GBinderClient* client);

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -125,7 +125,7 @@ gbinder_reader_read_buffer(
const void*
gbinder_reader_read_hidl_struct1(
GBinderReader* reader,
gsize size); /* since 1.0.9 */
gsize size); /* Since 1.0.9 */
#define gbinder_reader_read_hidl_struct(reader,type) \
((const type*)gbinder_reader_read_hidl_struct1(reader, sizeof(type)))
@@ -140,7 +140,7 @@ const void*
gbinder_reader_read_hidl_vec1(
GBinderReader* reader,
gsize* count,
guint expected_elemsize); /* since 1.0.9 */
guint expected_elemsize); /* Since 1.0.9 */
#define gbinder_reader_read_hidl_type_vec(reader,type,count) \
((const type*)gbinder_reader_read_hidl_vec1(reader, count, sizeof(type)))
@@ -152,6 +152,13 @@ gbinder_reader_read_hidl_string(
GBinderReader* reader)
G_GNUC_WARN_UNUSED_RESULT;
const char*
gbinder_reader_read_hidl_string_c(
GBinderReader* reader); /* Since 1.0.23 */
#define gbinder_reader_skip_hidl_string(reader) \
(gbinder_reader_read_hidl_string_c(reader) != NULL)
char**
gbinder_reader_read_hidl_string_vec(
GBinderReader* reader);
@@ -177,8 +184,13 @@ gbinder_reader_read_nullable_string16(
gboolean
gbinder_reader_read_nullable_string16_utf16(
GBinderReader* reader,
gunichar2** out,
gsize* len); /* since 1.0.17 */
const gunichar2** out,
gsize* len); /* Since 1.0.17 */
const gunichar2*
gbinder_reader_read_string16_utf16(
GBinderReader* reader,
gsize* len); /* Since 1.0.26 */
gboolean
gbinder_reader_skip_string16(
@@ -187,7 +199,7 @@ gbinder_reader_skip_string16(
const void*
gbinder_reader_read_byte_array(
GBinderReader* reader,
gsize* len); /* since 1.0.12 */
gsize* len); /* Since 1.0.12 */
gsize
gbinder_reader_bytes_read(

View File

@@ -67,6 +67,16 @@ gbinder_remote_request_copy_to_local(
GBinderRemoteRequest* req) /* since 1.0.6 */
G_GNUC_WARN_UNUSED_RESULT;
void
gbinder_remote_request_block(
GBinderRemoteRequest* req); /* Since 1.0.20 */
void
gbinder_remote_request_complete(
GBinderRemoteRequest* req,
GBinderLocalReply* reply,
int status); /* Since 1.0.20 */
/* Convenience function to decode requests with just one data item */
gboolean

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -37,6 +37,12 @@
G_BEGIN_DECLS
typedef
void
(*GBinderServiceManagerFunc)(
GBinderServiceManager* sm,
void* user_data);
/* GBinderServiceManagerListFunc callback returns TRUE to keep the services
* list, otherwise the caller will deallocate it. */
typedef
@@ -95,6 +101,15 @@ void
gbinder_servicemanager_unref(
GBinderServiceManager* sm);
gboolean
gbinder_servicemanager_is_present(
GBinderServiceManager* sm); /* Since 1.0.25 */
gboolean
gbinder_servicemanager_wait(
GBinderServiceManager* sm,
long max_wait_ms); /* Since 1.0.25 */
gulong
gbinder_servicemanager_list(
GBinderServiceManager* sm,
@@ -137,6 +152,12 @@ gbinder_servicemanager_cancel(
GBinderServiceManager* sm,
gulong id);
gulong
gbinder_servicemanager_add_presence_handler(
GBinderServiceManager* sm,
GBinderServiceManagerFunc func,
void* user_data); /* Since 1.0.25 */
gulong
gbinder_servicemanager_add_registration_handler(
GBinderServiceManager* sm,
@@ -149,6 +170,15 @@ gbinder_servicemanager_remove_handler(
GBinderServiceManager* sm,
gulong id); /* Since 1.0.13 */
void
gbinder_servicemanager_remove_handlers(
GBinderServiceManager* sm,
gulong* ids,
guint count); /* Since 1.0.25 */
#define gbinder_servicemanager_remove_all_handlers(r,ids) \
gbinder_servicemanager_remove_handlers(sm, ids, G_N_ELEMENTS(ids))
G_END_DECLS
#endif /* GBINDER_SERVICEMANAGER_H */

View File

@@ -0,0 +1,70 @@
/*
* Copyright (C) 2019 Jolla Ltd.
* Copyright (C) 2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the names of the copyright holders nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*/
#ifndef GBINDER_SERVICENAME_H
#define GBINDER_SERVICENAME_H
#include "gbinder_types.h"
G_BEGIN_DECLS
/* Since 1.0.26 */
struct gbinder_servicename {
const char* name;
};
GBinderServiceName*
gbinder_servicename_new(
GBinderServiceManager* sm,
GBinderLocalObject* object,
const char* name);
GBinderServiceName*
gbinder_servicename_ref(
GBinderServiceName* name);
void
gbinder_servicename_unref(
GBinderServiceName* name);
G_END_DECLS
#endif /* GBINDER_SERVICENAME_H */
/*
* Local Variables:
* mode: C
* c-basic-offset: 4
* indent-tabs-mode: nil
* End:
*/

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -13,9 +13,9 @@
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of Jolla Ltd nor the names of its contributors may
* be used to endorse or promote products derived from this software
* without specific prior written permission.
* 3. Neither the names of the copyright holders nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
@@ -68,6 +68,7 @@ typedef struct gbinder_reader GBinderReader;
typedef struct gbinder_remote_object GBinderRemoteObject;
typedef struct gbinder_remote_reply GBinderRemoteReply;
typedef struct gbinder_remote_request GBinderRemoteRequest;
typedef struct gbinder_servicename GBinderServiceName;
typedef struct gbinder_servicemanager GBinderServiceManager;
typedef struct gbinder_writer GBinderWriter;
typedef struct gbinder_parent GBinderParent;

View File

@@ -119,6 +119,16 @@ gbinder_writer_append_fd(
GBinderWriter* writer,
int fd); /* Since 1.0.18 */
gsize
gbinder_writer_bytes_written(
GBinderWriter* writer); /* since 1.0.21 */
void
gbinder_writer_overwrite_int32(
GBinderWriter* writer,
gsize offset,
gint32 value); /* since 1.0.21 */
guint
gbinder_writer_append_buffer_object_with_parent(
GBinderWriter* writer,
@@ -162,10 +172,38 @@ gbinder_writer_append_remote_object(
void
gbinder_writer_append_byte_array(
GBinderWriter* self,
GBinderWriter* writer,
const void* byte_array,
gint32 len); /* since 1.0.12 */
void*
gbinder_writer_malloc(
GBinderWriter* writer,
gsize size); /* since 1.0.19 */
void*
gbinder_writer_malloc0(
GBinderWriter* writer,
gsize size); /* since 1.0.19 */
#define gbinder_writer_new(writer,type) \
((type*) gbinder_writer_malloc(writer, sizeof(type)))
#define gbinder_writer_new0(writer,type) \
((type*) gbinder_writer_malloc0(writer, sizeof(type)))
void*
gbinder_writer_memdup(
GBinderWriter* writer,
const void* buf,
gsize size); /* since 1.0.19 */
void
gbinder_writer_add_cleanup(
GBinderWriter* writer,
GDestroyNotify destroy,
gpointer data); /* since 1.0.19 */
G_END_DECLS
#endif /* GBINDER_WRITER_H */

View File

@@ -1,5 +1,5 @@
Name: libgbinder
Version: 1.0.18
Version: 1.0.28
Release: 0
Summary: Binder client library
Group: Development/Libraries

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -13,9 +13,9 @@
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of Jolla Ltd nor the names of its contributors may
* be used to endorse or promote products derived from this software
* without specific prior written permission.
* 3. Neither the names of the copyright holders nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
@@ -176,6 +176,13 @@ gbinder_client_unref(
}
}
const char*
gbinder_client_interface(
GBinderClient* self) /* since 1.0.22 */
{
return G_LIKELY(self) ? gbinder_client_cast(self)->iface : NULL;
}
GBinderLocalRequest*
gbinder_client_new_request(
GBinderClient* self)

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -288,19 +288,15 @@ gbinder_driver_death_notification(
guint32 cmd,
GBinderRemoteObject* obj)
{
if (G_LIKELY(obj)) {
GBinderIoBuf write;
guint8 buf[4 + GBINDER_MAX_DEATH_NOTIFICATION_SIZE];
guint32* data = (guint32*)buf;
GBinderIoBuf write;
guint8 buf[4 + GBINDER_MAX_DEATH_NOTIFICATION_SIZE];
guint32* data = (guint32*)buf;
data[0] = cmd;
memset(&write, 0, sizeof(write));
write.ptr = (uintptr_t)buf;
write.size = 4 + self->io->encode_death_notification(data + 1, obj);
return gbinder_driver_write(self, &write) >= 0;
}
return FALSE;
data[0] = cmd;
memset(&write, 0, sizeof(write));
write.ptr = (uintptr_t)buf;
write.size = 4 + self->io->encode_death_notification(data + 1, obj);
return gbinder_driver_write(self, &write) >= 0;
}
static
@@ -458,7 +454,7 @@ gbinder_driver_handle_transaction(
&status);
break;
default:
GWARN("Unhandled transaction 0x%08x", tx.code);
GWARN("Unhandled transaction %s 0x%08x", iface, tx.code);
break;
}
@@ -535,6 +531,7 @@ gbinder_driver_handle_command(
} else if (cmd == io->br.transaction) {
gbinder_driver_handle_transaction(self, reg, handler, data);
} else if (cmd == io->br.dead_binder) {
guint8 buf[4 + GBINDER_MAX_PTR_COOKIE_SIZE];
guint64 handle = 0;
GBinderRemoteObject* obj;
@@ -545,6 +542,8 @@ gbinder_driver_handle_command(
gbinder_remote_object_handle_death_notification(obj);
gbinder_remote_object_unref(obj);
}
GVERBOSE("< BC_DEAD_BINDER_DONE %llu", (long long unsigned int)handle);
gbinder_driver_cmd_data(self, io->bc.dead_binder_done, data, buf);
} else if (cmd == io->br.clear_death_notification_done) {
GVERBOSE("> BR_CLEAR_DEATH_NOTIFICATION_DONE");
} else {
@@ -812,8 +811,13 @@ gbinder_driver_request_death_notification(
GBinderDriver* self,
GBinderRemoteObject* obj)
{
return gbinder_driver_death_notification
(self, self->io->bc.request_death_notification, obj);
if (G_LIKELY(obj)) {
GVERBOSE("< BC_REQUEST_DEATH_NOTIFICATION 0x%08x", obj->handle);
return gbinder_driver_death_notification(self,
self->io->bc.request_death_notification, obj);
} else {
return FALSE;
}
}
gboolean
@@ -821,8 +825,13 @@ gbinder_driver_clear_death_notification(
GBinderDriver* self,
GBinderRemoteObject* obj)
{
return gbinder_driver_death_notification
(self, self->io->bc.clear_death_notification, obj);
if (G_LIKELY(obj)) {
GVERBOSE("< BC_CLEAR_DEATH_NOTIFICATION 0x%08x", obj->handle);
return gbinder_driver_death_notification(self,
self->io->bc.clear_death_notification, obj);
} else {
return FALSE;
}
}
gboolean
@@ -1038,6 +1047,28 @@ gbinder_driver_transact(
return txstatus;
}
int
gbinder_driver_ping(
GBinderDriver* self,
GBinderObjectRegistry* reg,
guint32 handle)
{
const GBinderRpcProtocol* protocol = self->protocol;
GBinderLocalRequest* req = gbinder_local_request_new(self->io, NULL);
GBinderRemoteReply* reply = gbinder_remote_reply_new(reg);
GBinderWriter writer;
int ret;
gbinder_local_request_init_writer(req, &writer);
protocol->write_ping(&writer);
ret = gbinder_driver_transact(self, reg, handle, protocol->ping_tx,
req, reply);
gbinder_local_request_unref(req);
gbinder_remote_reply_unref(reply);
return ret;
}
GBinderLocalRequest*
gbinder_driver_local_request_new(
GBinderDriver* self,

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -131,6 +131,12 @@ gbinder_driver_transact(
GBinderLocalRequest* request,
GBinderRemoteReply* reply);
int
gbinder_driver_ping(
GBinderDriver* driver,
GBinderObjectRegistry* reg,
guint32 handle);
GBinderLocalRequest*
gbinder_driver_local_request_new(
GBinderDriver* self,

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -128,9 +128,13 @@ GBINDER_IO_FN(encode_local_object)(
struct flat_binder_object* dest = out;
memset(dest, 0, sizeof(*dest));
dest->hdr.type = obj ? BINDER_TYPE_BINDER : BINDER_TYPE_WEAK_HANDLE;
dest->flags = 0x7f | FLAT_BINDER_FLAG_ACCEPTS_FDS;
dest->binder = (uintptr_t)obj;
if (obj) {
dest->hdr.type = BINDER_TYPE_BINDER;
dest->flags = 0x7f | FLAT_BINDER_FLAG_ACCEPTS_FDS;
dest->binder = (uintptr_t)obj;
} else {
dest->hdr.type = BINDER_TYPE_WEAK_BINDER;
}
return sizeof(*dest);
}
@@ -409,7 +413,7 @@ guint
GBINDER_IO_FN(decode_buffer_object)(
GBinderBuffer* buf,
gsize offset,
GBinderBuffer** out)
GBinderIoBufferObject* out)
{
const void* data = (guint8*)buf->data + offset;
const gsize size = (offset < buf->size) ? (buf->size - offset) : 0;
@@ -417,12 +421,14 @@ GBINDER_IO_FN(decode_buffer_object)(
if (size >= sizeof(*flat) && flat->hdr.type == BINDER_TYPE_PTR) {
if (out) {
*out = gbinder_buffer_new_with_parent(buf,
(void*)(uintptr_t)flat->buffer, flat->length);
out->data = (void*)(uintptr_t)flat->buffer;
out->size = (gsize)flat->length;
out->parent_offset = (gsize)flat->parent_offset;
out->has_parent = (flat->flags & BINDER_BUFFER_FLAG_HAS_PARENT) ?
TRUE : FALSE;
}
return sizeof(*flat);
}
if (out) *out = NULL;
return 0;
}

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -43,6 +43,13 @@ typedef struct gbinder_io_buf {
gsize consumed;
} GBinderIoBuf;
typedef struct gbinder_io_buffer_object {
void* data;
gsize size;
gsize parent_offset;
gboolean has_parent;
} GBinderIoBufferObject;
typedef struct gbinder_io_tx_data {
int status;
guint32 code;
@@ -165,9 +172,9 @@ struct gbinder_io {
void* (*decode_binder_ptr_cookie)(const void* data);
guint (*decode_cookie)(const void* data, guint64* cookie);
guint (*decode_binder_object)(const void* data, gsize size,
GBinderObjectRegistry* reg, GBinderRemoteObject** obj);
GBinderObjectRegistry* reg, GBinderRemoteObject** obj);
guint (*decode_buffer_object)(GBinderBuffer* buf, gsize offset,
GBinderBuffer** out);
GBinderIoBufferObject* out);
guint (*decode_fd_object)(const void* data, gsize size, int* fd);
/* ioctl wrappers */

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -68,9 +68,9 @@ struct gbinder_ipc_priv {
GMutex local_objects_mutex;
GHashTable* local_objects;
/* We may need more loopers... But let's start with just one */
GMutex looper_mutex;
GBinderIpcLooper* looper;
GBinderIpcLooper* primary_loopers;
GBinderIpcLooper* blocked_loopers;
};
typedef GObjectClass GBinderIpcClass;
@@ -100,7 +100,7 @@ static pthread_mutex_t gbinder_ipc_mutex = PTHREAD_MUTEX_INITIALIZER;
* When the main thread receives GBinderIpcLooperTx:
*
* 1. Lets the object to process it and produce the response (GBinderOutput).
* 2. Writes one byte to the sending end of the tx pipe.
* 2. Writes one byte (TX_DONE) to the sending end of the tx pipe.
* 3. Unreferences GBinderIpcLooperTx
*
* When tx pipe wakes up the looper:
@@ -111,11 +111,26 @@ static pthread_mutex_t gbinder_ipc_mutex = PTHREAD_MUTEX_INITIALIZER;
* Note that GBinderIpcLooperTx can be deallocated on either looper or
* main thread, depending on whether looper gives up on the transaction
* before it gets processed.
*
* When transaction is blocked by gbinder_remote_request_block() call, it
* gets slightly more complicated. Then the main thread writes TX_BLOCKED
* to the pipe (rather than TX_DONE) and then looper thread spawn another
* looper and keeps waiting for TX_DONE.
*/
#define TX_DONE (0x2a)
#define TX_BLOCKED (0x3b)
typedef struct gbinder_ipc_looper_tx {
typedef enum gbinder_ipc_looper_tx_state {
GBINDER_IPC_LOOPER_TX_SCHEDULED,
GBINDER_IPC_LOOPER_TX_PROCESSING,
GBINDER_IPC_LOOPER_TX_PROCESSED,
GBINDER_IPC_LOOPER_TX_BLOCKING,
GBINDER_IPC_LOOPER_TX_BLOCKED,
GBINDER_IPC_LOOPER_TX_COMPLETE
} GBINDER_IPC_LOOPER_TX_STATE;
struct gbinder_ipc_looper_tx {
/* Reference count */
gint refcount;
/* These are filled by the looper: */
@@ -125,19 +140,23 @@ typedef struct gbinder_ipc_looper_tx {
GBinderLocalObject* obj;
GBinderRemoteRequest* req;
/* And these by the main thread processing the transaction: */
GBINDER_IPC_LOOPER_TX_STATE state;
GBinderLocalReply* reply;
int status;
} GBinderIpcLooperTx;
} /* GBinderIpcLooperTx */;
struct gbinder_ipc_looper {
gint refcount;
GBinderIpcLooper* next;
char* name;
GBinderHandler handler;
GBinderDriver* driver;
GBinderIpc* ipc; /* Not a reference! */
GThread* thread;
GMutex mutex;
GCond start_cond;
gboolean started;
gint exit;
gint started;
int pipefd[2];
int txfd[2];
};
@@ -178,6 +197,11 @@ typedef struct gbinder_ipc_tx_custom {
GBINDER_INLINE_FUNC const char* gbinder_ipc_name(GBinderIpc* self)
{ return gbinder_driver_dev(self->driver); }
static
GBinderIpcLooper*
gbinder_ipc_looper_new(
GBinderIpc* ipc);
/*==========================================================================*
* GBinderIpcLooperTx
*==========================================================================*/
@@ -245,6 +269,91 @@ gbinder_ipc_looper_tx_unref(
return dropped;
}
/*==========================================================================*
* State machine of transaction handling. All this is happening on the event
* thread and therefore doesn't need to be synchronized.
*
* SCHEDULED
* =========
* |
* PROCESSING
* ==========
* |
* --------------------- handler is called ---------------------------------
* |
* +---------------- request doesn't need to be blocked ----------+
* | |
* gbinder_remote_request_block() |
* | |
* BLOCKING -- gbinder_remote_request_complete() --> PROCESSED |
* ======== ========= |
* | | |
* --------------------- handler returns -----------------------------------
* | | |
* BLOCKED COMPLETE <-------+
* ======= ========
* ^
* ... |
* gbinder_remote_request_complete() is called later ----+
*==========================================================================*/
void
gbinder_remote_request_block(
GBinderRemoteRequest* req) /* Since 1.0.20 */
{
if (G_LIKELY(req)) {
GBinderIpcLooperTx* tx = req->tx;
GASSERT(tx);
if (G_LIKELY(tx)) {
GASSERT(tx->state == GBINDER_IPC_LOOPER_TX_PROCESSING);
if (tx->state == GBINDER_IPC_LOOPER_TX_PROCESSING) {
tx->state = GBINDER_IPC_LOOPER_TX_BLOCKING;
}
}
}
}
void
gbinder_remote_request_complete(
GBinderRemoteRequest* req,
GBinderLocalReply* reply,
int status) /* Since 1.0.20 */
{
if (G_LIKELY(req)) {
GBinderIpcLooperTx* tx = req->tx;
GASSERT(tx);
if (G_LIKELY(tx)) {
const guint8 done = TX_DONE;
switch (tx->state) {
case GBINDER_IPC_LOOPER_TX_BLOCKING:
/* Called by the transaction handler */
tx->status = status;
tx->reply = gbinder_local_reply_ref(reply);
tx->state = GBINDER_IPC_LOOPER_TX_PROCESSED;
break;
case GBINDER_IPC_LOOPER_TX_BLOCKED:
/* Really asynchronous completion */
tx->status = status;
tx->reply = gbinder_local_reply_ref(reply);
tx->state = GBINDER_IPC_LOOPER_TX_COMPLETE;
/* Wake up the looper */
(void)write(tx->pipefd[1], &done, sizeof(done));
break;
default:
GWARN("Unexpected state %d in request completion", tx->state);
break;
}
/* Clear the transaction reference */
gbinder_ipc_looper_tx_unref(tx, FALSE);
req->tx = NULL;
}
}
}
/*==========================================================================*
* GBinderIpcLooper
*==========================================================================*/
@@ -255,11 +364,70 @@ gbinder_ipc_looper_tx_handle(
gpointer data)
{
GBinderIpcLooperTx* tx = data;
guint8 done = TX_DONE;
GBinderRemoteRequest* req = tx->req;
GBinderLocalReply* reply;
int status = GBINDER_STATUS_OK;
guint8 done;
/*
* Transaction reference for gbinder_remote_request_block()
* and gbinder_remote_request_complete().
*/
req->tx = gbinder_ipc_looper_tx_ref(tx);
/* See state machine */
GASSERT(tx->state == GBINDER_IPC_LOOPER_TX_SCHEDULED);
tx->state = GBINDER_IPC_LOOPER_TX_PROCESSING;
/* Actually handle the transaction */
tx->reply = gbinder_local_object_handle_transaction(tx->obj, tx->req,
tx->code, tx->flags, &tx->status);
reply = gbinder_local_object_handle_transaction(tx->obj, req,
tx->code, tx->flags, &status);
/* Handle all possible return states */
switch (tx->state) {
case GBINDER_IPC_LOOPER_TX_PROCESSING:
/* Result was returned by the handler */
tx->reply = reply;
tx->status = status;
tx->state = GBINDER_IPC_LOOPER_TX_COMPLETE;
reply = NULL;
break;
case GBINDER_IPC_LOOPER_TX_PROCESSED:
/* Result has been provided to gbinder_remote_request_complete() */
tx->state = GBINDER_IPC_LOOPER_TX_COMPLETE;
break;
case GBINDER_IPC_LOOPER_TX_BLOCKING:
/* Result will be provided to gbinder_remote_request_complete() */
tx->state = GBINDER_IPC_LOOPER_TX_BLOCKED;
break;
default:
break;
}
/* In case handler returns a reply which it wasn't expected to return */
GASSERT(!reply);
gbinder_local_reply_unref(reply);
/* Drop the transaction reference unless blocked */
if (tx->state == GBINDER_IPC_LOOPER_TX_BLOCKED) {
done = TX_BLOCKED;
/*
* From this point on, it's GBinderRemoteRequest who's holding
* reference to GBinderIpcLooperTx, not the other way around and
* not both ways. Even if gbinder_remote_request_complete() never
* gets called, transaction will still be completed when the last
* reference to GBinderRemoteRequest goes away. And if request
* never gets deallocated... oh well.
*/
gbinder_remote_request_unref(tx->req);
tx->req = NULL;
} else {
done = TX_DONE;
if (req->tx) {
gbinder_ipc_looper_tx_unref(req->tx, FALSE);
req->tx = NULL;
}
}
/* And wake up the looper */
(void)write(tx->pipefd[1], &done, sizeof(done));
@@ -274,6 +442,52 @@ gbinder_ipc_looper_tx_done(
gbinder_ipc_looper_tx_unref(data, FALSE);
}
static
gboolean
gbinder_ipc_looper_remove_from_list(
GBinderIpcLooper* looper,
GBinderIpcLooper** list)
{
/* Caller holds looper_mutex */
if (*list) {
if ((*list) == looper) {
(*list) = looper->next;
looper->next = NULL;
return TRUE;
} else {
GBinderIpcLooper* prev = (*list);
while (prev->next) {
if (prev->next == looper) {
prev->next = looper->next;
looper->next = NULL;
return TRUE;
}
prev = prev->next;
}
}
}
return FALSE;
}
static
gboolean
gbinder_ipc_looper_remove_primary(
GBinderIpcLooper* looper)
{
return gbinder_ipc_looper_remove_from_list(looper,
&looper->ipc->priv->primary_loopers);
}
static
gboolean
gbinder_ipc_looper_remove_blocked(
GBinderIpcLooper* looper)
{
return gbinder_ipc_looper_remove_from_list(looper,
&looper->ipc->priv->blocked_loopers);
}
static
GBinderLocalReply*
gbinder_ipc_looper_transact(
@@ -318,11 +532,65 @@ gbinder_ipc_looper_transact(
if ((fds[1].revents & POLLIN) &&
read(fds[1].fd, &done, sizeof(done)) == 1) {
/* Normal completion */
if (done == TX_BLOCKED) {
/*
* We are going to block this looper for potentially
* significant period of time. Start new looper to
* accept normal incoming requests and terminate this
* one when we are done with this transaction.
*
* For the duration of the transaction, this looper is
* moved to the blocked_loopers list.
*/
GBinderIpcPriv* priv = looper->ipc->priv;
/* Lock */
g_mutex_lock(&priv->looper_mutex);
if (gbinder_ipc_looper_remove_primary(looper)) {
GBinderIpcLooper* new_looper;
GDEBUG("Primary looper %s is blocked", looper->name);
looper->next = priv->blocked_loopers;
priv->blocked_loopers = looper;
/* Looper will exit once transaction completes */
g_atomic_int_set(&looper->exit, 1);
/* Create new primary looper to replace this one */
new_looper = gbinder_ipc_looper_new(ipc);
if (new_looper) {
new_looper->next = priv->primary_loopers;
priv->primary_loopers = new_looper;
}
}
g_mutex_unlock(&priv->looper_mutex);
/* Unlock */
/* Block until asynchronous transaction gets completed. */
done = 0;
memset(fds, 0, sizeof(fds));
fds[0].fd = looper->pipefd[0];
fds[0].events = POLLIN | POLLERR | POLLHUP | POLLNVAL;
fds[1].fd = tx->pipefd[0];
fds[1].events = POLLIN | POLLERR | POLLHUP | POLLNVAL;
poll(fds, 2, -1);
if ((fds[1].revents & POLLIN) &&
read(fds[1].fd, &done, sizeof(done)) == 1) {
GASSERT(done == TX_DONE);
}
}
}
if (done) {
GASSERT(done == TX_DONE);
reply = gbinder_local_reply_ref(tx->reply);
status = tx->status;
if (!gbinder_ipc_looper_tx_unref(tx, TRUE)) {
/* gbinder_ipc_looper_tx_free() will close those */
/*
* This wasn't the last references meaning that
* gbinder_ipc_looper_tx_free() will close the
* descriptors and we will have to create a new
* pipe for the next transaction.
*/
looper->txfd[0] = looper->txfd[1] = -1;
}
} else {
@@ -348,6 +616,7 @@ gbinder_ipc_looper_free(
close(looper->txfd[1]);
}
gbinder_driver_unref(looper->driver);
g_free(looper->name);
g_cond_clear(&looper->start_cond);
g_mutex_clear(&looper->mutex);
g_slice_free(GBinderIpcLooper, looper);
@@ -384,11 +653,11 @@ gbinder_ipc_looper_thread(
if (gbinder_driver_enter_looper(driver)) {
struct pollfd pipefd;
int result;
int res;
GDEBUG("Looper %s running", gbinder_driver_dev(driver));
GDEBUG("Looper %s running", looper->name);
g_mutex_lock(&looper->mutex);
looper->started = TRUE;
g_atomic_int_set(&looper->started, TRUE);
g_cond_broadcast(&looper->start_cond);
g_mutex_unlock(&looper->mutex);
@@ -396,31 +665,33 @@ gbinder_ipc_looper_thread(
pipefd.fd = looper->pipefd[0]; /* read end of the pipe */
pipefd.events = POLLIN | POLLERR | POLLHUP | POLLNVAL;
result = gbinder_driver_poll(driver, &pipefd);
while (looper->ipc && ((result & POLLIN) || !result)) {
if (result & POLLIN) {
/* No need to synchronize access to looper->ipc because
res = gbinder_driver_poll(driver, &pipefd);
while (!g_atomic_int_get(&looper->exit) && ((res & POLLIN) || !res)) {
if (res & POLLIN) {
/*
* No need to synchronize access to looper->ipc because
* the other thread would wait until this thread exits
* before setting looper->ipc to NULL */
* before setting looper->ipc to NULL.
*/
GBinderIpc* ipc = gbinder_ipc_ref(looper->ipc);
GBinderObjectRegistry* reg = gbinder_ipc_object_registry(ipc);
/* But that gbinder_driver_read() may unref GBinderIpc */
int ret = gbinder_driver_read(driver, reg, &looper->handler);
/* And this gbinder_ipc_unref() may release the last ref: */
gbinder_ipc_unref(ipc);
/* And at this point looper->ipc may be NULL */
if (ret < 0) {
GDEBUG("Looper %s failed", gbinder_driver_dev(driver));
GDEBUG("Looper %s failed", looper->name);
break;
}
}
if (pipefd.revents) {
/* Any event from this pipe terminates the loop */
GDEBUG("Looper %s is asked to exit",
gbinder_driver_dev(driver));
/* Any event from this pipe terminates the loop */
if (pipefd.revents || g_atomic_int_get(&looper->exit)) {
GDEBUG("Looper %s is requested to exit", looper->name);
break;
}
result = gbinder_driver_poll(driver, &pipefd);
res = gbinder_driver_poll(driver, &pipefd);
}
gbinder_driver_exit_looper(driver);
@@ -432,24 +703,26 @@ gbinder_ipc_looper_thread(
*/
if (looper->ipc) {
GBinderIpcPriv* priv = looper->ipc->priv;
/* Lock */
g_mutex_lock(&priv->looper_mutex);
if (priv->looper == looper) {
if (gbinder_ipc_looper_remove_blocked(looper) ||
gbinder_ipc_looper_remove_primary(looper)) {
/* Spontaneous exit */
priv->looper = NULL;
GDEBUG("Looper %s exits", gbinder_driver_dev(driver));
GDEBUG("Looper %s exits", looper->name);
gbinder_ipc_looper_unref(looper);
} else {
/* Main thread is shutting it down */
GDEBUG("Looper %s done", gbinder_driver_dev(driver));
GDEBUG("Looper %s done", looper->name);
}
g_mutex_unlock(&priv->looper_mutex);
/* Unlock */
} else {
GDEBUG("Looper %s is abandoned", gbinder_driver_dev(driver));
GDEBUG("Looper %s is abandoned", looper->name);
}
} else {
g_mutex_lock(&looper->mutex);
looper->started = TRUE;
g_atomic_int_set(&looper->started, TRUE);
g_cond_broadcast(&looper->start_cond);
g_mutex_unlock(&looper->mutex);
}
@@ -472,20 +745,24 @@ gbinder_ipc_looper_new(
};
GError* error = NULL;
GBinderIpcLooper* looper = g_slice_new0(GBinderIpcLooper);
static gint gbinder_ipc_next_looper_id = 1;
guint id = (guint)g_atomic_int_add(&gbinder_ipc_next_looper_id, 1);
memcpy(looper->pipefd, fd, sizeof(fd));
looper->txfd[0] = looper->txfd[1] = -1;
g_atomic_int_set(&looper->refcount, 1);
g_cond_init(&looper->start_cond);
g_mutex_init(&looper->mutex);
looper->name = g_strdup_printf("%s#%u", gbinder_ipc_name(ipc), id);
looper->handler.f = &handler_functions;
looper->ipc = ipc;
looper->driver = gbinder_driver_ref(ipc->driver);
looper->thread = g_thread_try_new(gbinder_ipc_name(ipc),
looper->thread = g_thread_try_new(looper->name,
gbinder_ipc_looper_thread, looper, &error);
if (looper->thread) {
/* gbinder_ipc_looper_thread() will release this reference: */
gbinder_ipc_looper_ref(looper);
GDEBUG("Starting looper %s", looper->name);
return looper;
} else {
GERR("Failed to create looper thread: %s", GERRMSG(error));
@@ -505,29 +782,29 @@ gbinder_ipc_looper_check(
if (G_LIKELY(self)) {
GBinderIpcPriv* priv = self->priv;
if (!priv->looper) {
if (!priv->primary_loopers) {
GBinderIpcLooper* looper;
/* Lock */
g_mutex_lock(&priv->looper_mutex);
if (!priv->looper) {
GDEBUG("Starting looper %s", gbinder_ipc_name(self));
priv->looper = gbinder_ipc_looper_new(self);
looper = priv->primary_loopers;
if (!looper) {
looper = priv->primary_loopers = gbinder_ipc_looper_new(self);
}
g_mutex_unlock(&priv->looper_mutex);
/* Unlock */
/* We are not ready to accept incoming transactions until
* looper has started. We may need to wait a bit. */
looper = priv->looper;
if (!looper->started) {
if (looper && !g_atomic_int_get(&looper->started)) {
/* Lock */
g_mutex_lock(&looper->mutex);
if (!looper->started) {
if (!g_atomic_int_get(&looper->started)) {
g_cond_wait_until(&looper->start_cond, &looper->mutex,
g_get_monotonic_time() +
GBINDER_IPC_LOOPER_START_TIMEOUT_SEC *
G_TIME_SPAN_SECOND);
GASSERT(looper->started);
GASSERT(g_atomic_int_get(&looper->started));
}
g_mutex_unlock(&looper->mutex);
/* Unlock */
@@ -536,6 +813,54 @@ gbinder_ipc_looper_check(
}
}
static
void
gbinder_ipc_looper_stop(
GBinderIpcLooper* looper)
{
/* Caller checks looper for NULL */
if (looper->thread && looper->thread != g_thread_self()) {
guint8 done = TX_DONE;
GDEBUG("Stopping looper %s", looper->name);
g_atomic_int_set(&looper->exit, TRUE);
if (write(looper->pipefd[1], &done, sizeof(done)) <= 0) {
looper->thread = NULL;
}
}
}
static
GBinderIpcLooper*
gbinder_ipc_looper_stop_all(
GBinderIpcLooper* loopers,
GBinderIpcLooper* list)
{
while (list) {
GBinderIpcLooper* looper = list;
GBinderIpcLooper* next = looper->next;
gbinder_ipc_looper_stop(looper);
looper->next = loopers;
loopers = looper;
list = next;
}
return loopers;
}
static
void
gbinder_ipc_looper_join(
GBinderIpcLooper* looper)
{
/* Caller checks looper for NULL */
if (looper->thread && looper->thread != g_thread_self()) {
g_thread_join(looper->thread);
looper->thread = NULL;
}
looper->ipc = NULL;
}
/*==========================================================================*
* GBinderObjectRegistry
*==========================================================================*/
@@ -661,7 +986,8 @@ static
GBinderRemoteObject*
gbinder_ipc_priv_get_remote_object(
GBinderIpcPriv* priv,
guint32 handle)
guint32 handle,
gboolean maybe_dead)
{
GBinderRemoteObject* obj = NULL;
void* key = GINT_TO_POINTER(handle);
@@ -674,7 +1000,11 @@ gbinder_ipc_priv_get_remote_object(
if (obj) {
gbinder_remote_object_ref(obj);
} else {
obj = gbinder_remote_object_new(priv->self, handle);
/*
* If maybe_dead is TRUE, the caller is supposed to try reanimating
* the object on the main thread not holding any global locks.
*/
obj = gbinder_remote_object_new(priv->self, handle, maybe_dead);
if (!priv->remote_objects) {
priv->remote_objects = g_hash_table_new
(g_direct_hash, g_direct_equal);
@@ -690,10 +1020,11 @@ gbinder_ipc_priv_get_remote_object(
GBinderRemoteObject*
gbinder_ipc_get_remote_object(
GBinderIpc* self,
guint32 handle)
guint32 handle,
gboolean maybe_dead)
{
/* GBinderServiceManager makes sure that GBinderIpc pointer is not NULL */
return gbinder_ipc_priv_get_remote_object(self->priv, handle);
return gbinder_ipc_priv_get_remote_object(self->priv, handle, maybe_dead);
}
GBINDER_INLINE_FUNC
@@ -745,7 +1076,7 @@ gbinder_ipc_object_registry_get_remote(
guint32 handle)
{
return gbinder_ipc_priv_get_remote_object
(gbinder_ipc_priv_from_object_registry(reg), handle);
(gbinder_ipc_priv_from_object_registry(reg), handle, FALSE);
}
/*==========================================================================*
@@ -839,14 +1170,19 @@ gbinder_ipc_tx_internal_exec(
GBinderObjectRegistry* reg = &self->priv->object_registry;
/* Perform synchronous transaction */
tx->reply = gbinder_remote_reply_new(&self->priv->object_registry);
tx->status = gbinder_driver_transact(self->driver, reg,
tx->handle, tx->code, tx->req, tx->reply);
if (tx->status != GBINDER_STATUS_OK &&
gbinder_remote_reply_is_empty(tx->reply)) {
/* Drop useless reply */
gbinder_remote_reply_unref(tx->reply);
tx->reply = NULL;
if (tx->flags & GBINDER_TX_FLAG_ONEWAY) {
tx->status = gbinder_driver_transact(self->driver, reg, tx->handle,
tx->code, tx->req, NULL);
} else {
tx->reply = gbinder_remote_reply_new(&self->priv->object_registry);
tx->status = gbinder_driver_transact(self->driver, reg, tx->handle,
tx->code, tx->req, tx->reply);
if (tx->status != GBINDER_STATUS_OK &&
gbinder_remote_reply_is_empty(tx->reply)) {
/* Drop useless reply */
gbinder_remote_reply_unref(tx->reply);
tx->reply = NULL;
}
}
}
@@ -1232,7 +1568,7 @@ gbinder_ipc_dispose(
{
GBinderIpc* self = GBINDER_IPC(object);
GBinderIpcPriv* priv = self->priv;
GBinderIpcLooper* looper;
GBinderIpcLooper* loopers = NULL;
GVERBOSE_("%s", self->dev);
/* Lock */
@@ -1253,22 +1589,19 @@ gbinder_ipc_dispose(
/* Lock */
g_mutex_lock(&priv->looper_mutex);
looper = priv->looper;
priv->looper = NULL;
loopers = gbinder_ipc_looper_stop_all(loopers, priv->primary_loopers);
loopers = gbinder_ipc_looper_stop_all(loopers, priv->blocked_loopers);
priv->blocked_loopers = NULL;
priv->primary_loopers = NULL;
g_mutex_unlock(&priv->looper_mutex);
/* Unlock */
if (looper) {
if (looper->thread && looper->thread != g_thread_self()) {
guint8 done = TX_DONE;
while (loopers) {
GBinderIpcLooper* looper = loopers;
GDEBUG("Stopping looper %s", gbinder_ipc_name(looper->ipc));
if (write(looper->pipefd[1], &done, sizeof(done)) > 0) {
g_thread_join(looper->thread);
looper->thread = NULL;
}
}
looper->ipc = NULL;
loopers = looper->next;
looper->next = NULL;
gbinder_ipc_looper_join(looper);
gbinder_ipc_looper_unref(looper);
}

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -99,7 +99,8 @@ gbinder_ipc_new_local_object(
GBinderRemoteObject*
gbinder_ipc_get_remote_object(
GBinderIpc* ipc,
guint32 handle);
guint32 handle,
gboolean maybe_dead);
GBinderRemoteReply*
gbinder_ipc_transact_sync_reply(

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -13,9 +13,9 @@
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of Jolla Ltd nor the names of its contributors may
* be used to endorse or promote products derived from this software
* without specific prior written permission.
* 3. Neither the names of the copyright holders nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
@@ -53,6 +53,13 @@ G_DEFINE_TYPE(GBinderLocalObject, gbinder_local_object, G_TYPE_OBJECT)
G_TYPE_INSTANCE_GET_CLASS((obj), GBINDER_TYPE_LOCAL_OBJECT, \
GBinderLocalObjectClass)
typedef
GBinderLocalReply*
(*GBinderLocalObjectTxHandler)(
GBinderLocalObject* self,
GBinderRemoteRequest* req,
int* status);
enum gbinder_local_object_signal {
SIGNAL_WEAK_REFS_CHANGED,
SIGNAL_STRONG_REFS_CHANGED,
@@ -78,6 +85,9 @@ gbinder_local_object_default_can_handle_transaction(
guint code)
{
switch (code) {
case GBINDER_PING_TRANSACTION:
case GBINDER_INTERFACE_TRANSACTION:
return GBINDER_LOCAL_TRANSACTION_LOOPER;
case HIDL_PING_TRANSACTION:
case HIDL_GET_DESCRIPTOR_TRANSACTION:
case HIDL_DESCRIPTOR_CHAIN_TRANSACTION:
@@ -110,6 +120,38 @@ gbinder_local_object_default_handle_transaction(
}
}
static
GBinderLocalReply*
gbinder_local_object_ping_transaction(
GBinderLocalObject* self,
GBinderRemoteRequest* req,
int* status)
{
const GBinderIo* io = gbinder_local_object_io(self);
GBinderLocalReply* reply = gbinder_local_reply_new(io);
GVERBOSE(" PING_TRANSACTION");
gbinder_local_reply_append_int32(reply, GBINDER_STATUS_OK);
*status = GBINDER_STATUS_OK;
return reply;
}
static
GBinderLocalReply*
gbinder_local_object_interface_transaction(
GBinderLocalObject* self,
GBinderRemoteRequest* req,
int* status)
{
const GBinderIo* io = gbinder_local_object_io(self);
GBinderLocalReply* reply = gbinder_local_reply_new(io);
GVERBOSE(" INTERFACE_TRANSACTION");
gbinder_local_reply_append_string16(reply, self->iface);
*status = GBINDER_STATUS_OK;
return reply;
}
static
GBinderLocalReply*
gbinder_local_object_hidl_ping_transaction(
@@ -120,12 +162,10 @@ gbinder_local_object_hidl_ping_transaction(
/*android.hidl.base@1.0::IBase interfaceDescriptor() */
const GBinderIo* io = gbinder_local_object_io(self);
GBinderLocalReply* reply = gbinder_local_reply_new(io);
GBinderWriter writer;
GVERBOSE(" HIDL_PING_TRANSACTION \"%s\"",
gbinder_remote_request_interface(req));
gbinder_local_reply_init_writer(reply, &writer);
gbinder_writer_append_int32(&writer, GBINDER_STATUS_OK);
gbinder_local_reply_append_int32(reply, GBINDER_STATUS_OK);
*status = GBINDER_STATUS_OK;
return reply;
}
@@ -187,23 +227,31 @@ gbinder_local_object_default_handle_looper_transaction(
guint flags,
int* status)
{
GBinderLocalObjectTxHandler handler = NULL;
switch (code) {
case GBINDER_PING_TRANSACTION:
handler = gbinder_local_object_ping_transaction;
break;
case GBINDER_INTERFACE_TRANSACTION:
handler = gbinder_local_object_interface_transaction;
break;
case HIDL_PING_TRANSACTION:
GASSERT(!(flags & GBINDER_TX_FLAG_ONEWAY));
return gbinder_local_object_hidl_ping_transaction
(self, req, status);
handler = gbinder_local_object_hidl_ping_transaction;
break;
case HIDL_GET_DESCRIPTOR_TRANSACTION:
GASSERT(!(flags & GBINDER_TX_FLAG_ONEWAY));
return gbinder_local_object_hidl_get_descriptor_transaction
(self, req, status);
handler = gbinder_local_object_hidl_get_descriptor_transaction;
break;
case HIDL_DESCRIPTOR_CHAIN_TRANSACTION:
GASSERT(!(flags & GBINDER_TX_FLAG_ONEWAY));
return gbinder_local_object_hidl_descriptor_chain_transaction
(self, req, status);
handler = gbinder_local_object_hidl_descriptor_chain_transaction;
break;
default:
if (status) *status = (-EBADMSG);
return NULL;
}
GASSERT(!(flags & GBINDER_TX_FLAG_ONEWAY));
return handler(self, req, status);
}
static
@@ -286,7 +334,7 @@ gbinder_local_object_new(
GBinderLocalTransactFunc txproc,
void* user_data)
{
/* Should only be called from gbinder_ipc_new_local_local_object() */
/* Should only be called from gbinder_ipc_new_local_object() */
if (G_LIKELY(ipc)) {
GBinderLocalObject* self = g_object_new
(GBINDER_TYPE_LOCAL_OBJECT, NULL);

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -320,9 +320,9 @@ gbinder_reader_read_object(
static
gboolean
gbinder_reader_read_buffer_impl(
gbinder_reader_read_buffer_object(
GBinderReader* reader,
GBinderBuffer** out)
GBinderIoBufferObject* out)
{
GBinderReaderPriv* p = gbinder_reader_cast(reader);
@@ -339,7 +339,6 @@ gbinder_reader_read_buffer_impl(
return TRUE;
}
}
if (out) *out = NULL;
return FALSE;
}
@@ -347,34 +346,36 @@ GBinderBuffer*
gbinder_reader_read_buffer(
GBinderReader* reader)
{
GBinderBuffer* buf = NULL;
GBinderIoBufferObject obj;
gbinder_reader_read_buffer_impl(reader, &buf);
return buf;
if (gbinder_reader_read_buffer_object(reader, &obj)) {
const GBinderReaderData* data = gbinder_reader_cast(reader)->data;
GBinderBuffer* buf = data->buffer;
return gbinder_buffer_new_with_parent(buf, obj.data, obj.size);
}
return NULL;
}
gboolean
gbinder_reader_skip_buffer(
GBinderReader* reader)
{
return gbinder_reader_read_buffer_impl(reader, NULL);
return gbinder_reader_read_buffer_object(reader, NULL);
}
/* Helper for gbinder_reader_read_hidl_struct() macro */
const void*
gbinder_reader_read_hidl_struct1(
GBinderReader* reader,
gsize size) /* since 1.0.9 */
gsize size) /* Since 1.0.9 */
{
const void* result = NULL;
GBinderBuffer* buf = gbinder_reader_read_buffer(reader);
GBinderIoBufferObject obj;
/* Check the size */
if (buf && buf->size == size) {
result = buf->data;
if (gbinder_reader_read_buffer_object(reader, &obj) && obj.size == size) {
return obj.data;
}
gbinder_buffer_free(buf);
return result;
return NULL;
}
/* Doesn't copy the data */
@@ -384,30 +385,28 @@ gbinder_reader_read_hidl_vec(
gsize* count,
gsize* elemsize)
{
GBinderBuffer* buf = gbinder_reader_read_buffer(reader);
gsize out_count = 0, out_elemsize = 0;
GBinderIoBufferObject obj;
const void* out = NULL;
gsize out_count = 0, out_elemsize = 0;
if (buf && buf->size == sizeof(GBinderHidlVec)) {
const GBinderHidlVec* vec = buf->data;
if (gbinder_reader_read_buffer_object(reader, &obj) &&
obj.data && obj.size == sizeof(GBinderHidlVec)) {
const GBinderHidlVec* vec = obj.data;
const void* next = vec->data.ptr;
if (next) {
GBinderBuffer* vbuf = gbinder_reader_read_buffer(reader);
if (vbuf && vbuf->data == next && ((!vec->count && !vbuf->size) ||
(vec->count && vbuf->size && !(vbuf->size % vec->count)))) {
out_elemsize = vec->count ? (vbuf->size / vec->count) : 0;
if (gbinder_reader_read_buffer_object(reader, &obj) &&
obj.data == next && ((!vec->count && !obj.size) ||
(vec->count && obj.size && !(obj.size % vec->count)))) {
out_elemsize = vec->count ? (obj.size / vec->count) : 0;
out_count = vec->count;
out = vbuf->data;
out = obj.data;
}
gbinder_buffer_free(vbuf);
} else if (!vec->count) {
/* Any non-NULL pointer just to indicate success */
/* Any non-NULL pointer just to indicate success? */
out = vec;
}
}
gbinder_buffer_free(buf);
if (elemsize) {
*elemsize = out_elemsize;
}
@@ -422,7 +421,7 @@ const void*
gbinder_reader_read_hidl_vec1(
GBinderReader* reader,
gsize* count,
guint expected_elem_size) /* since 1.0.9 */
guint expected_elem_size) /* Since 1.0.9 */
{
gsize actual;
const void* data = gbinder_reader_read_hidl_vec(reader, count, &actual);
@@ -431,89 +430,94 @@ gbinder_reader_read_hidl_vec1(
return (data && (actual == expected_elem_size || !actual)) ? data : NULL;
}
const char*
gbinder_reader_read_hidl_string_c(
GBinderReader* reader) /* Since 1.0.23 */
{
GBinderIoBufferObject obj;
if (gbinder_reader_read_buffer_object(reader, &obj) &&
obj.data && obj.size == sizeof(GBinderHidlString)) {
const GBinderHidlString* str = obj.data;
if (gbinder_reader_read_buffer_object(reader, &obj) &&
obj.has_parent &&
obj.parent_offset == GBINDER_HIDL_STRING_BUFFER_OFFSET &&
obj.data == str->data.str &&
obj.size == str->len + 1 &&
str->data.str[str->len] == 0) {
return str->data.str;
}
}
return NULL;
}
char*
gbinder_reader_read_hidl_string(
GBinderReader* reader)
{
GBinderBuffer* buf = gbinder_reader_read_buffer(reader);
char* str = NULL;
if (buf && buf->size == sizeof(GBinderHidlString)) {
const GBinderHidlString* s = buf->data;
GBinderBuffer* sbuf = gbinder_reader_read_buffer(reader);
if (sbuf && sbuf->size == s->len + 1 &&
sbuf->data == s->data.str &&
s->data.str[s->len] == 0) {
str = g_strdup(s->data.str);
}
gbinder_buffer_free(sbuf);
}
gbinder_buffer_free(buf);
return str;
/* This function should've been called gbinder_reader_dup_hidl_string */
return g_strdup(gbinder_reader_read_hidl_string_c(reader));
}
char**
gbinder_reader_read_hidl_string_vec(
GBinderReader* reader)
{
GBinderBuffer* buf = gbinder_reader_read_buffer(reader);
GBinderIoBufferObject obj;
/* First buffer contains hidl_vector */
if (buf && buf->size == sizeof(GBinderHidlVec)) {
GBinderHidlVec* vec = buf->data;
if (gbinder_reader_read_buffer_object(reader, &obj) &&
obj.data && obj.size == sizeof(GBinderHidlVec)) {
GBinderHidlVec* vec = obj.data;
const guint n = vec->count;
const void* next = vec->data.ptr;
gbinder_buffer_free(buf);
if (!next && !n) {
char** out = g_new(char*, 1);
/* Should this be considered an error? */
return g_new0(char*, 1);
} else if (gbinder_reader_read_buffer_object(reader, &obj) &&
/* The second buffer (if any) contains n hidl_string's */
obj.parent_offset == GBINDER_HIDL_VEC_BUFFER_OFFSET &&
obj.has_parent &&
obj.data == next &&
obj.size == (sizeof(GBinderHidlString) * n)) {
const GBinderHidlString* strings = obj.data;
GPtrArray* list = g_ptr_array_sized_new(n + 1);
guint i;
out[0] = NULL;
return out;
} else {
/* The second buffer (if any) contains n hidl_string's */
buf = gbinder_reader_read_buffer(reader);
if (buf && buf->data == next &&
buf->size == (sizeof(GBinderHidlString) * n)) {
const GBinderHidlString* strings = buf->data;
GBinderBuffer* sbuf;
GPtrArray* list = g_ptr_array_new();
guint i;
/* Now we expect n buffers containing the actual data */
for (i = 0; i < n &&
gbinder_reader_read_buffer_object(reader, &obj); i++) {
const GBinderHidlString* s = strings + i;
const gsize expected_offset = (i * sizeof(*s)) +
GBINDER_HIDL_STRING_BUFFER_OFFSET;
if (obj.has_parent &&
obj.parent_offset == expected_offset &&
obj.data == s->data.str &&
obj.size == s->len + 1 &&
s->data.str[s->len] == 0) {
char* name = g_strdup(s->data.str);
/* Now we expect n buffers containing the actual data */
for (i=0; i<n &&
(sbuf = gbinder_reader_read_buffer(reader)); i++) {
const GBinderHidlString* s = strings + i;
if (sbuf->size == s->len + 1 &&
sbuf->data == s->data.str &&
s->data.str[s->len] == 0) {
char* name = g_strdup(s->data.str);
g_ptr_array_add(list, name);
GVERBOSE_("%u. %s", i + 1, name);
gbinder_buffer_free(sbuf);
} else {
GWARN("Unexpected hidl_string buffer %p/%u vs %p/%u",
sbuf->data, (guint)sbuf->size, s->data.str, s->len);
gbinder_buffer_free(sbuf);
break;
}
g_ptr_array_add(list, name);
GVERBOSE_("%u. %s", i + 1, name);
} else {
GWARN("Unexpected hidl_string buffer %p/%u vs %p/%u",
obj.data, (guint)obj.size, s->data.str, s->len);
break;
}
if (i == n) {
gbinder_buffer_free(buf);
g_ptr_array_add(list, NULL);
return (char**)g_ptr_array_free(list, FALSE);
}
g_ptr_array_set_free_func(list, g_free);
g_ptr_array_free(list, TRUE);
}
if (i == n) {
g_ptr_array_add(list, NULL);
return (char**)g_ptr_array_free(list, FALSE);
}
g_ptr_array_set_free_func(list, g_free);
g_ptr_array_free(list, TRUE);
}
}
GWARN("Invalid hidl_vec<string>");
gbinder_buffer_free(buf);
return NULL;
}
@@ -546,7 +550,7 @@ gbinder_reader_read_nullable_string16(
GBinderReader* reader,
char** out)
{
gunichar2* str;
const gunichar2* str;
gsize len;
if (gbinder_reader_read_nullable_string16_utf16(reader, &str, &len)) {
@@ -561,8 +565,8 @@ gbinder_reader_read_nullable_string16(
gboolean
gbinder_reader_read_nullable_string16_utf16(
GBinderReader* reader,
gunichar2** out,
gsize* out_len) /* since 1.0.17 */
const gunichar2** out,
gsize* out_len) /* Since 1.0.17 */
{
GBinderReaderPriv* p = gbinder_reader_cast(reader);
@@ -582,7 +586,7 @@ gbinder_reader_read_nullable_string16_utf16(
return TRUE;
} else if (len >= 0) {
const guint32 padded_len = G_ALIGN4((len + 1)*2);
gunichar2* utf16 = (gunichar2*)(p->ptr + 4);
const gunichar2* utf16 = (gunichar2*)(p->ptr + 4);
if ((p->ptr + padded_len + 4) <= p->end) {
p->ptr += padded_len + 4;
@@ -599,6 +603,21 @@ gbinder_reader_read_nullable_string16_utf16(
return FALSE;
}
const gunichar2*
gbinder_reader_read_string16_utf16(
GBinderReader* reader,
gsize* len) /* Since 1.0.26 */
{
const gunichar2* str;
/*
* Use gbinder_reader_read_nullable_string16_utf16 to distinguish
* NULL string from a parsing failure.
*/
return gbinder_reader_read_nullable_string16_utf16(reader, &str, len) ?
str : NULL;
}
char*
gbinder_reader_read_string16(
GBinderReader* reader)
@@ -638,7 +657,7 @@ gbinder_reader_skip_string16(
const void*
gbinder_reader_read_byte_array(
GBinderReader* reader,
gsize* len) /* since 1.0.12 */
gsize* len) /* Since 1.0.12 */
{
GBinderReaderPriv* p = gbinder_reader_cast(reader);
const void* data = NULL;

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -13,9 +13,9 @@
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of Jolla Ltd nor the names of its contributors may
* be used to endorse or promote products derived from this software
* without specific prior written permission.
* 3. Neither the names of the copyright holders nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
@@ -65,9 +65,16 @@ void
gbinder_remote_object_died_on_main_thread(
GBinderRemoteObject* self)
{
GBinderIpc* ipc = self->ipc;
GBinderDriver* driver = ipc->driver;
GASSERT(!self->dead);
self->dead = TRUE;
g_signal_emit(self, gbinder_remote_object_signals[SIGNAL_DEATH], 0);
if (!self->dead) {
self->dead = TRUE;
gbinder_driver_clear_death_notification(driver, self);
gbinder_driver_release(driver, self->handle);
g_signal_emit(self, gbinder_remote_object_signals[SIGNAL_DEATH], 0);
}
}
static
@@ -79,6 +86,46 @@ gbinder_remote_object_died_handle(
return G_SOURCE_REMOVE;
}
/*==========================================================================*
* Internal interface
*==========================================================================*/
gboolean
gbinder_remote_object_reanimate(
GBinderRemoteObject* self)
{
/*
* Don't try to reanimate those who hasn't died yet. Reanimation is
* kind of a special case and should only be used for servicemanager
* objects.
*/
if (self->dead) {
GBinderIpc* ipc = self->ipc;
GBinderObjectRegistry* reg = gbinder_ipc_object_registry(ipc);
/* Kick the horse */
if (gbinder_driver_ping(ipc->driver, reg, self->handle) == 0) {
/* Wow, it's alive! */
self->dead = FALSE;
gbinder_driver_acquire(ipc->driver, self->handle);
gbinder_driver_request_death_notification(ipc->driver, self);
}
}
return !self->dead;
}
void
gbinder_remote_object_handle_death_notification(
GBinderRemoteObject* self)
{
/* This function is invoked from the looper thread, the caller has
* checked the object pointer */
GVERBOSE_("%p %u", self, self->handle);
g_main_context_invoke_full(self->priv->context, G_PRIORITY_DEFAULT,
gbinder_remote_object_died_handle, gbinder_remote_object_ref(self),
g_object_unref);
}
/*==========================================================================*
* Interface
*==========================================================================*/
@@ -86,15 +133,19 @@ gbinder_remote_object_died_handle(
GBinderRemoteObject*
gbinder_remote_object_new(
GBinderIpc* ipc,
guint32 handle)
guint32 handle,
gboolean dead)
{
if (G_LIKELY(ipc) && gbinder_driver_acquire(ipc->driver, handle)) {
if (G_LIKELY(ipc)) {
GBinderRemoteObject* self = g_object_new
(GBINDER_TYPE_REMOTE_OBJECT, NULL);
self->ipc = gbinder_ipc_ref(ipc);
self->handle = handle;
gbinder_driver_request_death_notification(ipc->driver, self);
if (!(self->dead = dead)) {
gbinder_driver_acquire(ipc->driver, handle);
gbinder_driver_request_death_notification(ipc->driver, self);
}
return self;
}
return NULL;
@@ -152,18 +203,6 @@ gbinder_remote_object_remove_handler(
}
}
void
gbinder_remote_object_handle_death_notification(
GBinderRemoteObject* self)
{
/* This function is invoked from the looper thread, the caller has
* checked the object pointer */
GVERBOSE_("%p %u", self, self->handle);
g_main_context_invoke_full(self->priv->context, G_PRIORITY_DEFAULT,
gbinder_remote_object_died_handle, gbinder_remote_object_ref(self),
g_object_unref);
}
/*==========================================================================*
* Internals
*==========================================================================*/
@@ -200,8 +239,10 @@ gbinder_remote_object_finalize(
GBinderIpc* ipc = self->ipc;
GBinderDriver* driver = ipc->driver;
gbinder_driver_clear_death_notification(driver, self);
gbinder_driver_release(driver, self->handle);
if (!self->dead) {
gbinder_driver_clear_death_notification(driver, self);
gbinder_driver_release(driver, self->handle);
}
gbinder_ipc_unref(ipc);
G_OBJECT_CLASS(gbinder_remote_object_parent_class)->finalize(remote);
}

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -13,9 +13,9 @@
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of Jolla Ltd nor the names of its contributors may
* be used to endorse or promote products derived from this software
* without specific prior written permission.
* 3. Neither the names of the copyright holders nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
@@ -54,7 +54,12 @@ struct gbinder_remote_object {
GBinderRemoteObject*
gbinder_remote_object_new(
GBinderIpc* ipc,
guint32 handle);
guint32 handle,
gboolean maybe_dead);
gboolean
gbinder_remote_object_reanimate(
GBinderRemoteObject* obj);
void
gbinder_remote_object_handle_death_notification(

View File

@@ -40,7 +40,10 @@
#include <gutil_macros.h>
struct gbinder_remote_request {
#include <errno.h>
typedef struct gbinder_remote_request_priv {
GBinderRemoteRequest pub;
gint refcount;
pid_t pid;
uid_t euid;
@@ -49,7 +52,11 @@ struct gbinder_remote_request {
char* iface2;
gsize header_size;
GBinderReaderData data;
};
} GBinderRemoteRequestPriv;
GBINDER_INLINE_FUNC GBinderRemoteRequestPriv*
gbinder_remote_request_cast(GBinderRemoteRequest* pub)
{ return G_LIKELY(pub) ? G_CAST(pub,GBinderRemoteRequestPriv,pub) : NULL; }
GBinderRemoteRequest*
gbinder_remote_request_new(
@@ -58,7 +65,7 @@ gbinder_remote_request_new(
pid_t pid,
uid_t euid)
{
GBinderRemoteRequest* self = g_slice_new0(GBinderRemoteRequest);
GBinderRemoteRequestPriv* self = g_slice_new0(GBinderRemoteRequestPriv);
GBinderReaderData* data = &self->data;
g_atomic_int_set(&self->refcount, 1);
@@ -66,13 +73,15 @@ gbinder_remote_request_new(
self->euid = euid;
self->protocol = protocol;
data->reg = gbinder_object_registry_ref(reg);
return self;
return &self->pub;
}
GBinderLocalRequest*
gbinder_remote_request_copy_to_local(
GBinderRemoteRequest* self)
GBinderRemoteRequest* req)
{
GBinderRemoteRequestPriv* self = gbinder_remote_request_cast(req);
if (G_LIKELY(self)) {
GBinderReaderData* d = &self->data;
@@ -84,21 +93,27 @@ gbinder_remote_request_copy_to_local(
static
void
gbinder_remote_request_free(
GBinderRemoteRequest* self)
GBinderRemoteRequestPriv* self)
{
GBinderReaderData* data = &self->data;
GBinderRemoteRequest* req = &self->pub;
GASSERT(!req->tx);
if (req->tx) {
GWARN("Request is dropped without completing the transaction");
gbinder_remote_request_complete(req, NULL, -ECANCELED);
}
gbinder_object_registry_unref(data->reg);
gbinder_buffer_free(data->buffer);
g_free(self->iface2);
g_slice_free(GBinderRemoteRequest, self);
g_slice_free(GBinderRemoteRequestPriv, self);
}
static
inline
void
gbinder_remote_request_init_reader2(
GBinderRemoteRequest* self,
GBinderRemoteRequestPriv* self,
GBinderReader* p)
{
/* The caller has already checked the request for NULL */
@@ -115,10 +130,12 @@ gbinder_remote_request_init_reader2(
void
gbinder_remote_request_set_data(
GBinderRemoteRequest* self,
GBinderRemoteRequest* req,
guint32 txcode,
GBinderBuffer* buffer)
{
GBinderRemoteRequestPriv* self = gbinder_remote_request_cast(req);
if (G_LIKELY(self)) {
GBinderReaderData* data = &self->data;
GBinderReader reader;
@@ -145,26 +162,32 @@ gbinder_remote_request_set_data(
const char*
gbinder_remote_request_interface(
GBinderRemoteRequest* self)
GBinderRemoteRequest* req)
{
GBinderRemoteRequestPriv* self = gbinder_remote_request_cast(req);
return G_LIKELY(self) ? self->iface : NULL;
}
GBinderRemoteRequest*
gbinder_remote_request_ref(
GBinderRemoteRequest* self)
GBinderRemoteRequest* req)
{
GBinderRemoteRequestPriv* self = gbinder_remote_request_cast(req);
if (G_LIKELY(self)) {
GASSERT(self->refcount > 0);
g_atomic_int_inc(&self->refcount);
}
return self;
return req;
}
void
gbinder_remote_request_unref(
GBinderRemoteRequest* self)
GBinderRemoteRequest* req)
{
GBinderRemoteRequestPriv* self = gbinder_remote_request_cast(req);
if (G_LIKELY(self)) {
GASSERT(self->refcount > 0);
if (g_atomic_int_dec_and_test(&self->refcount)) {
@@ -175,9 +198,11 @@ gbinder_remote_request_unref(
void
gbinder_remote_request_init_reader(
GBinderRemoteRequest* self,
GBinderRemoteRequest* req,
GBinderReader* reader)
{
GBinderRemoteRequestPriv* self = gbinder_remote_request_cast(req);
if (G_LIKELY(self)) {
gbinder_remote_request_init_reader2(self, reader);
} else {
@@ -187,15 +212,19 @@ gbinder_remote_request_init_reader(
pid_t
gbinder_remote_request_sender_pid(
GBinderRemoteRequest* self)
GBinderRemoteRequest* req)
{
GBinderRemoteRequestPriv* self = gbinder_remote_request_cast(req);
return G_LIKELY(self) ? self->pid : (uid_t)(-1);
}
uid_t
gbinder_remote_request_sender_euid(
GBinderRemoteRequest* self)
GBinderRemoteRequest* req)
{
GBinderRemoteRequestPriv* self = gbinder_remote_request_cast(req);
return G_LIKELY(self) ? self->euid : (uid_t)(-1);
}
@@ -209,9 +238,11 @@ gbinder_remote_request_read_int32(
gboolean
gbinder_remote_request_read_uint32(
GBinderRemoteRequest* self,
GBinderRemoteRequest* req,
guint32* value)
{
GBinderRemoteRequestPriv* self = gbinder_remote_request_cast(req);
if (G_LIKELY(self)) {
GBinderReader reader;
@@ -231,9 +262,11 @@ gbinder_remote_request_read_int64(
gboolean
gbinder_remote_request_read_uint64(
GBinderRemoteRequest* self,
GBinderRemoteRequest* req,
guint64* value)
{
GBinderRemoteRequestPriv* self = gbinder_remote_request_cast(req);
if (G_LIKELY(self)) {
GBinderReader reader;
@@ -245,8 +278,10 @@ gbinder_remote_request_read_uint64(
const char*
gbinder_remote_request_read_string8(
GBinderRemoteRequest* self)
GBinderRemoteRequest* req)
{
GBinderRemoteRequestPriv* self = gbinder_remote_request_cast(req);
if (G_LIKELY(self)) {
GBinderReader reader;
@@ -258,8 +293,10 @@ gbinder_remote_request_read_string8(
char*
gbinder_remote_request_read_string16(
GBinderRemoteRequest* self)
GBinderRemoteRequest* req)
{
GBinderRemoteRequestPriv* self = gbinder_remote_request_cast(req);
if (G_LIKELY(self)) {
GBinderReader reader;
@@ -271,8 +308,10 @@ gbinder_remote_request_read_string16(
GBinderRemoteObject*
gbinder_remote_request_read_object(
GBinderRemoteRequest* self)
GBinderRemoteRequest* req)
{
GBinderRemoteRequestPriv* self = gbinder_remote_request_cast(req);
if (G_LIKELY(self)) {
GBinderReader reader;

View File

@@ -37,6 +37,10 @@
#include "gbinder_types_p.h"
struct gbinder_remote_request {
GBinderIpcLooperTx* tx;
};
GBinderRemoteRequest*
gbinder_remote_request_new(
GBinderObjectRegistry* reg,

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -50,6 +50,14 @@
#define STRICT_MODE_PENALTY_GATHER (0x40 << 16)
#define BINDER_RPC_FLAGS (STRICT_MODE_PENALTY_GATHER)
static
void
gbinder_rpc_protocol_binder_write_ping(
GBinderWriter* writer)
{
/* No payload */
}
static
void
gbinder_rpc_protocol_binder_write_rpc_header(
@@ -99,6 +107,15 @@ gbinder_rpc_protocol_hwbinder_write_rpc_header(
gbinder_writer_append_string8(writer, iface);
}
static
void
gbinder_rpc_protocol_hwbinder_write_ping(
GBinderWriter* writer)
{
gbinder_rpc_protocol_hwbinder_write_rpc_header(writer,
"android.hidl.base@1.0::IBase");
}
static
const char*
gbinder_rpc_protocol_hwbinder_read_rpc_header(
@@ -115,13 +132,17 @@ gbinder_rpc_protocol_hwbinder_read_rpc_header(
*==========================================================================*/
const GBinderRpcProtocol gbinder_rpc_protocol_binder = {
.read_rpc_header = gbinder_rpc_protocol_binder_read_rpc_header,
.write_rpc_header = gbinder_rpc_protocol_binder_write_rpc_header
.ping_tx = GBINDER_PING_TRANSACTION,
.write_ping = gbinder_rpc_protocol_binder_write_ping,
.write_rpc_header = gbinder_rpc_protocol_binder_write_rpc_header,
.read_rpc_header = gbinder_rpc_protocol_binder_read_rpc_header
};
const GBinderRpcProtocol gbinder_rpc_protocol_hwbinder = {
.read_rpc_header = gbinder_rpc_protocol_hwbinder_read_rpc_header,
.write_rpc_header = gbinder_rpc_protocol_hwbinder_write_rpc_header
.ping_tx = HIDL_PING_TRANSACTION,
.write_ping = gbinder_rpc_protocol_hwbinder_write_ping,
.write_rpc_header = gbinder_rpc_protocol_hwbinder_write_rpc_header,
.read_rpc_header = gbinder_rpc_protocol_hwbinder_read_rpc_header
};
const GBinderRpcProtocol*

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -42,9 +42,11 @@
*/
struct gbinder_rpc_protocol {
guint32 ping_tx;
void (*write_ping)(GBinderWriter* writer);
void (*write_rpc_header)(GBinderWriter* writer, const char* iface);
const char* (*read_rpc_header)(GBinderReader* reader, guint32 txcode,
char** iface);
void (*write_rpc_header)(GBinderWriter* writer, const char* iface);
};
extern const GBinderRpcProtocol gbinder_rpc_protocol_binder;

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -41,9 +41,14 @@
#include <gbinder_client.h>
#include <gutil_idlepool.h>
#include <gutil_misc.h>
#include <errno.h>
#define PRESENSE_WAIT_MS_MIN (100)
#define PRESENSE_WAIT_MS_MAX (1000)
#define PRESENSE_WAIT_MS_STEP (100)
typedef struct gbinder_servicemanager_watch {
char* name;
char* detail;
@@ -53,6 +58,10 @@ typedef struct gbinder_servicemanager_watch {
struct gbinder_servicemanager_priv {
GHashTable* watch_table;
gulong death_id;
gboolean present;
guint presence_check_id;
guint presence_check_delay_ms;
};
G_DEFINE_ABSTRACT_TYPE(GBinderServiceManager, gbinder_servicemanager,
@@ -72,10 +81,12 @@ G_DEFINE_ABSTRACT_TYPE(GBinderServiceManager, gbinder_servicemanager,
G_TYPE_CHECK_CLASS_TYPE(klass, GBINDER_TYPE_SERVICEMANAGER)
enum gbinder_servicemanager_signal {
SIGNAL_PRESENCE,
SIGNAL_REGISTRATION,
SIGNAL_COUNT
};
static const char SIGNAL_PRESENCE_NAME[] = "servicemanager-presence";
static const char SIGNAL_REGISTRATION_NAME[] = "servicemanager-registration";
#define DETAIL_LEN 32
@@ -253,6 +264,129 @@ gbinder_servicemanager_add_service_tx_free(
g_slice_free(GBinderServiceManagerAddServiceTxData, data);
}
static
void
gbinder_servicemanager_reanimated(
GBinderServiceManager* self)
{
GBinderServiceManagerPriv* priv = self->priv;
if (priv->presence_check_id) {
g_source_remove(priv->presence_check_id);
priv->presence_check_id = 0;
}
GINFO("Service manager %s has appeared", self->dev);
/* Re-arm the watches */
if (g_hash_table_size(priv->watch_table) > 0) {
gpointer value;
GHashTableIter it;
GBinderServiceManagerClass* klass =
GBINDER_SERVICEMANAGER_GET_CLASS(self);
g_hash_table_iter_init(&it, priv->watch_table);
while (g_hash_table_iter_next(&it, NULL, &value)) {
GBinderServiceManagerWatch* watch = value;
GASSERT(!watch->watched);
watch->watched = klass->watch(self, watch->name);
if (watch->watched) {
GDEBUG("Watching %s", watch->name);
} else {
GWARN("Failed to watch %s", watch->name);
}
}
}
g_signal_emit(self, gbinder_servicemanager_signals[SIGNAL_PRESENCE], 0);
}
static
gboolean
gbinder_servicemanager_presense_check_timer(
gpointer user_data)
{
GBinderServiceManager* self = GBINDER_SERVICEMANAGER(user_data);
GBinderRemoteObject* remote = self->client->remote;
GBinderServiceManagerPriv* priv = self->priv;
gboolean result;
GASSERT(remote->dead);
gbinder_servicemanager_ref(self);
if (gbinder_remote_object_reanimate(remote)) {
/* Done */
priv->presence_check_id = 0;
gbinder_servicemanager_reanimated(self);
result = G_SOURCE_REMOVE;
} else if (priv->presence_check_delay_ms < PRESENSE_WAIT_MS_MAX) {
priv->presence_check_delay_ms += PRESENSE_WAIT_MS_STEP;
priv->presence_check_id = g_timeout_add(priv->presence_check_delay_ms,
gbinder_servicemanager_presense_check_timer, self);
result = G_SOURCE_REMOVE;
} else {
result = G_SOURCE_CONTINUE;
}
gbinder_servicemanager_unref(self);
return result;
}
static
void
gbinder_servicemanager_presence_check_start(
GBinderServiceManager* self)
{
GBinderServiceManagerPriv* priv = self->priv;
GASSERT(!priv->presence_check_id);
priv->presence_check_delay_ms = PRESENSE_WAIT_MS_MIN;
priv->presence_check_id = g_timeout_add(PRESENSE_WAIT_MS_MIN,
gbinder_servicemanager_presense_check_timer, self);
}
static
void
gbinder_servicemanager_died(
GBinderRemoteObject* remote,
void* user_data)
{
GBinderServiceManager* self = GBINDER_SERVICEMANAGER(user_data);
GBinderServiceManagerPriv* priv = self->priv;
GWARN("Service manager %s has died", self->dev);
gbinder_servicemanager_presence_check_start(self);
/* Will re-arm watches after servicemanager gets restarted */
if (g_hash_table_size(priv->watch_table) > 0) {
gpointer value;
GHashTableIter it;
GBinderServiceManagerClass* klass =
GBINDER_SERVICEMANAGER_GET_CLASS(self);
g_hash_table_iter_init(&it, priv->watch_table);
while (g_hash_table_iter_next(&it, NULL, &value)) {
GBinderServiceManagerWatch* watch = value;
if (watch->watched) {
GDEBUG("Unwatching %s", watch->name);
watch->watched = FALSE;
klass->unwatch(self, watch->name);
}
}
}
g_signal_emit(self, gbinder_servicemanager_signals[SIGNAL_PRESENCE], 0);
}
static
void
gbinder_servicemanager_sleep_ms(
gulong ms)
{
struct timespec wait;
wait.tv_sec = ms/1000; /* seconds */
wait.tv_nsec = (ms % 1000) * 1000000; /* nanoseconds */
while (nanosleep(&wait, &wait) == -1 && errno == EINTR &&
(wait.tv_sec > 0 || wait.tv_nsec > 0));
}
/*==========================================================================*
* Internal interface
*==========================================================================*/
@@ -271,21 +405,25 @@ gbinder_servicemanager_new_with_type(
if (!dev) dev = klass->default_device;
ipc = gbinder_ipc_new(dev, klass->rpc_protocol);
if (ipc) {
/* Create a possible dead remote object */
GBinderRemoteObject* object = gbinder_ipc_get_remote_object
(ipc, klass->handle);
(ipc, klass->handle, TRUE);
if (object) {
gboolean first_ref;
/* Lock */
g_mutex_lock(&klass->mutex);
if (klass->table) {
self = g_hash_table_lookup(klass->table, dev);
}
if (self) {
first_ref = FALSE;
gbinder_servicemanager_ref(self);
} else {
char* key = g_strdup(dev); /* Owned by the hashtable */
GVERBOSE_("%s", dev);
first_ref = TRUE;
self = g_object_new(type, NULL);
self->client = gbinder_client_new(object, klass->iface);
self->dev = gbinder_remote_object_dev(object);
@@ -297,6 +435,20 @@ gbinder_servicemanager_new_with_type(
}
g_mutex_unlock(&klass->mutex);
/* Unlock */
if (first_ref) {
GBinderServiceManagerPriv* priv = self->priv;
priv->death_id =
gbinder_remote_object_add_death_handler(object,
gbinder_servicemanager_died, self);
/* Query the actual state if necessary */
gbinder_remote_object_reanimate(object);
if (object->dead) {
gbinder_servicemanager_presence_check_start(self);
}
GDEBUG("%s has %sservice manager", dev,
object->dead ? "no " : "");
}
gbinder_remote_object_unref(object);
}
gbinder_ipc_unref(ipc);
@@ -384,6 +536,58 @@ gbinder_servicemanager_unref(
}
}
gboolean
gbinder_servicemanager_is_present(
GBinderServiceManager* self) /* Since 1.0.25 */
{
return G_LIKELY(self) && !self->client->remote->dead;
}
gboolean
gbinder_servicemanager_wait(
GBinderServiceManager* self,
long max_wait_ms) /* Since 1.0.25 */
{
if (G_LIKELY(self)) {
GBinderRemoteObject* remote = self->client->remote;
if (!remote->dead) {
return TRUE;
} else if (gbinder_remote_object_reanimate(remote)) {
gbinder_servicemanager_reanimated(self);
return TRUE;
} else if (max_wait_ms != 0) {
/* Zero timeout means a singe check and it's already done */
long delay_ms = PRESENSE_WAIT_MS_MIN;
while (max_wait_ms != 0) {
if (max_wait_ms > 0) {
if (max_wait_ms < delay_ms) {
delay_ms = max_wait_ms;
max_wait_ms = 0;
} else {
max_wait_ms -= delay_ms;
}
}
gbinder_servicemanager_sleep_ms(delay_ms);
if (gbinder_remote_object_reanimate(remote)) {
gbinder_servicemanager_reanimated(self);
return TRUE;
}
if (delay_ms < PRESENSE_WAIT_MS_MAX) {
delay_ms += PRESENSE_WAIT_MS_STEP;
if (delay_ms > PRESENSE_WAIT_MS_MAX) {
delay_ms = PRESENSE_WAIT_MS_MAX;
}
}
}
/* Timeout */
GWARN("Timeout waiting for service manager %s", self->dev);
}
}
return FALSE;
}
gulong
gbinder_servicemanager_list(
GBinderServiceManager* self,
@@ -513,6 +717,16 @@ gbinder_servicemanager_cancel(
}
}
gulong
gbinder_servicemanager_add_presence_handler(
GBinderServiceManager* self,
GBinderServiceManagerFunc func,
void* user_data) /* Since 1.0.25 */
{
return (G_LIKELY(self) && G_LIKELY(func)) ? g_signal_connect(self,
SIGNAL_PRESENCE_NAME, G_CALLBACK(func), user_data) : 0;
}
gulong
gbinder_servicemanager_add_registration_handler(
GBinderServiceManager* self,
@@ -546,8 +760,8 @@ gbinder_servicemanager_add_registration_handler(
watch = gbinder_servicemanager_watch_new(name);
g_hash_table_insert(priv->watch_table, watch->name, watch);
}
if (!watch->watched) {
watch->watched = klass->watch(self, name);
if (!watch->watched && !self->client->remote->dead) {
watch->watched = klass->watch(self, watch->name);
if (watch->watched) {
GDEBUG("Watching %s", watch->name);
} else {
@@ -570,26 +784,46 @@ gbinder_servicemanager_remove_handler(
GBinderServiceManager* self,
gulong id) /* Since 1.0.13 */
{
if (G_LIKELY(self) && G_LIKELY(id)) {
GBinderServiceManagerClass* klass =
GBINDER_SERVICEMANAGER_GET_CLASS(self);
GBinderServiceManagerPriv* priv = self->priv;
GHashTableIter it;
gpointer value;
gbinder_servicemanager_remove_handlers(self, &id, 1);
}
g_signal_handler_disconnect(self, id);
g_hash_table_iter_init(&it, priv->watch_table);
while (g_hash_table_iter_next(&it, NULL, &value)) {
GBinderServiceManagerWatch* watch = value;
void
gbinder_servicemanager_remove_handlers(
GBinderServiceManager* self,
gulong* ids,
guint count) /* Since 1.0.25 */
{
if (G_LIKELY(self) && G_LIKELY(ids) && G_LIKELY(count)) {
guint i, disconnected = 0;
if (watch->watched && !g_signal_has_handler_pending(self,
gbinder_servicemanager_signals[SIGNAL_REGISTRATION],
watch->quark, TRUE)) {
/* This must be the one we have just removed */
GDEBUG("Unwatching %s", watch->name);
watch->watched = FALSE;
klass->unwatch(self, watch->name);
break;
for (i = 0; i < count; i++) {
if (ids[i]) {
g_signal_handler_disconnect(self, ids[i]);
disconnected++;
ids[i] = 0;
}
}
if (disconnected) {
GBinderServiceManagerClass* klass =
GBINDER_SERVICEMANAGER_GET_CLASS(self);
GBinderServiceManagerPriv* priv = self->priv;
GHashTableIter it;
gpointer value;
g_hash_table_iter_init(&it, priv->watch_table);
while (disconnected && g_hash_table_iter_next(&it, NULL, &value)) {
GBinderServiceManagerWatch* watch = value;
if (watch->watched && !g_signal_has_handler_pending(self,
gbinder_servicemanager_signals[SIGNAL_REGISTRATION],
watch->quark, TRUE)) {
/* This must be one of those we have just removed */
GDEBUG("Unwatching %s", watch->name);
watch->watched = FALSE;
klass->unwatch(self, watch->name);
disconnected--;
}
}
}
}
@@ -662,6 +896,10 @@ gbinder_servicemanager_finalize(
GBinderServiceManager* self = GBINDER_SERVICEMANAGER(object);
GBinderServiceManagerPriv* priv = self->priv;
if (priv->presence_check_id) {
g_source_remove(priv->presence_check_id);
}
gbinder_remote_object_remove_handler(self->client->remote, priv->death_id);
g_hash_table_destroy(priv->watch_table);
gutil_idle_pool_destroy(self->pool);
gbinder_client_unref(self->client);
@@ -674,13 +912,18 @@ gbinder_servicemanager_class_init(
GBinderServiceManagerClass* klass)
{
GObjectClass* object_class = G_OBJECT_CLASS(klass);
GType type = G_OBJECT_CLASS_TYPE(klass);
g_mutex_init(&klass->mutex);
g_type_class_add_private(klass, sizeof(GBinderServiceManagerPriv));
object_class->dispose = gbinder_servicemanager_dispose;
object_class->finalize = gbinder_servicemanager_finalize;
gbinder_servicemanager_signals[SIGNAL_PRESENCE] =
g_signal_new(SIGNAL_PRESENCE_NAME, type,
G_SIGNAL_RUN_FIRST, 0, NULL, NULL, NULL,
G_TYPE_NONE, 0);
gbinder_servicemanager_signals[SIGNAL_REGISTRATION] =
g_signal_new(SIGNAL_REGISTRATION_NAME, G_OBJECT_CLASS_TYPE(klass),
g_signal_new(SIGNAL_REGISTRATION_NAME, type,
G_SIGNAL_RUN_FIRST | G_SIGNAL_DETAILED, 0, NULL, NULL, NULL,
G_TYPE_NONE, 1, G_TYPE_STRING);
}

173
src/gbinder_servicename.c Normal file
View File

@@ -0,0 +1,173 @@
/*
* Copyright (C) 2019 Jolla Ltd.
* Copyright (C) 2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the names of the copyright holders nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "gbinder_types_p.h"
#include "gbinder_servicename.h"
#include "gbinder_servicemanager.h"
#include "gbinder_local_object.h"
#include "gbinder_log.h"
#include <gutil_macros.h>
/* Since 1.0.26 */
typedef struct gbinder_servicename_priv {
GBinderServiceName pub;
gint refcount;
char* name;
GBinderLocalObject* object;
GBinderServiceManager* sm;
gulong presence_id;
gulong add_call_id;
} GBinderServiceNamePriv;
GBINDER_INLINE_FUNC GBinderServiceNamePriv*
gbinder_servicename_cast(GBinderServiceName* pub)
{ return G_CAST(pub, GBinderServiceNamePriv, pub); }
/*==========================================================================*
* Implementation
*==========================================================================*/
static
void
gbinder_servicename_add_service_done(
GBinderServiceManager* sm,
int status,
void* user_data)
{
GBinderServiceNamePriv* priv = user_data;
GASSERT(priv->add_call_id);
priv->add_call_id = 0;
if (status) {
GWARN("Error %d adding name \"%s\"", status, priv->name);
} else {
GDEBUG("Service \"%s\" has been registered", priv->name);
}
}
static
void
gbinder_servicename_add_service(
GBinderServiceNamePriv* priv)
{
GDEBUG("Adding service \"%s\"", priv->name);
gbinder_servicemanager_cancel(priv->sm, priv->add_call_id);
priv->add_call_id = gbinder_servicemanager_add_service(priv->sm,
priv->name, priv->object, gbinder_servicename_add_service_done, priv);
}
static
void
gbinder_servicename_presence_handler(
GBinderServiceManager* sm,
void* user_data)
{
GBinderServiceNamePriv* priv = user_data;
if (gbinder_servicemanager_is_present(sm)) {
gbinder_servicename_add_service(priv);
} else if (priv->add_call_id) {
gbinder_servicemanager_cancel(priv->sm, priv->add_call_id);
priv->add_call_id = 0;
}
}
/*==========================================================================*
* Interface
*==========================================================================*/
GBinderServiceName*
gbinder_servicename_new(
GBinderServiceManager* sm,
GBinderLocalObject* object,
const char* name)
{
if (G_LIKELY(sm) && G_LIKELY(object) && G_LIKELY(name)) {
GBinderServiceNamePriv* priv = g_slice_new0(GBinderServiceNamePriv);
GBinderServiceName* self = &priv->pub;
g_atomic_int_set(&priv->refcount, 1);
priv->object = gbinder_local_object_ref(object);
priv->sm = gbinder_servicemanager_ref(sm);
self->name = priv->name = g_strdup(name);
priv->presence_id = gbinder_servicemanager_add_presence_handler(sm,
gbinder_servicename_presence_handler, priv);
if (gbinder_servicemanager_is_present(sm)) {
gbinder_servicename_add_service(priv);
}
return self;
} else {
return NULL;
}
}
GBinderServiceName*
gbinder_servicename_ref(
GBinderServiceName* self)
{
if (G_LIKELY(self)) {
GBinderServiceNamePriv* priv = gbinder_servicename_cast(self);
GASSERT(priv->refcount > 0);
g_atomic_int_inc(&priv->refcount);
}
return self;
}
void
gbinder_servicename_unref(
GBinderServiceName* self)
{
if (G_LIKELY(self)) {
GBinderServiceNamePriv* priv = gbinder_servicename_cast(self);
GASSERT(priv->refcount > 0);
if (g_atomic_int_dec_and_test(&priv->refcount)) {
gbinder_servicemanager_cancel(priv->sm, priv->add_call_id);
gbinder_servicemanager_remove_handler(priv->sm, priv->presence_id);
gbinder_servicemanager_unref(priv->sm);
gbinder_local_object_unref(priv->object);
g_free(priv->name);
g_slice_free(GBinderServiceName, self);
}
}
}
/*
* Local Variables:
* mode: C
* c-basic-offset: 4
* indent-tabs-mode: nil
* End:
*/

View File

@@ -45,6 +45,7 @@ typedef struct gbinder_object_registry GBinderObjectRegistry;
typedef struct gbinder_output_data GBinderOutputData;
typedef struct gbinder_rpc_protocol GBinderRpcProtocol;
typedef struct gbinder_servicepoll GBinderServicePoll;
typedef struct gbinder_ipc_looper_tx GBinderIpcLooperTx;
#define GBINDER_INLINE_FUNC static inline

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -149,6 +149,14 @@ gbinder_writer_init(
gbinder_writer_cast(self)->data = data;
}
gsize
gbinder_writer_bytes_written(
GBinderWriter* self) /* since 1.0.21 */
{
GBinderWriterData* data = gbinder_writer_data(self);
return data->bytes->len;
}
void
gbinder_writer_append_bool(
GBinderWriter* self,
@@ -199,6 +207,26 @@ gbinder_writer_data_append_int32(
*ptr = value;
}
void
gbinder_writer_overwrite_int32(
GBinderWriter* self,
gsize offset,
gint32 value) /* since 1.0.21 */
{
GBinderWriterData* data = gbinder_writer_data(self);
if (G_LIKELY(data)) {
GByteArray* buf = data->bytes;
if (buf->len >= offset + sizeof(gint32)) {
*((gint32*)(buf->data + offset)) = value;
} else {
GWARN("Can't overwrite at %lu as buffer is only %u bytes long",
(gulong)offset, buf->len);
}
}
}
void
gbinder_writer_append_int64(
GBinderWriter* self,
@@ -889,6 +917,73 @@ gbinder_writer_data_append_remote_object(
gbinder_writer_data_record_offset(data, offset);
}
static
void*
gbinder_writer_alloc(
GBinderWriter* self,
gsize size,
gpointer (*alloc)(gsize),
void (*dealloc)())
{
GBinderWriterData* data = gbinder_writer_data(self);
if (G_LIKELY(data)) {
void* ptr = alloc(size);
data->cleanup = gbinder_cleanup_add(data->cleanup, dealloc, ptr);
return ptr;
}
return NULL;
}
void*
gbinder_writer_malloc(
GBinderWriter* self,
gsize size) /* since 1.0.19 */
{
return gbinder_writer_alloc(self, size, g_malloc, g_free);
}
void*
gbinder_writer_malloc0(
GBinderWriter* self,
gsize size) /* since 1.0.19 */
{
return gbinder_writer_alloc(self, size, g_malloc0, g_free);
}
void*
gbinder_writer_memdup(
GBinderWriter* self,
const void* buf,
gsize size) /* since 1.0.19 */
{
if (buf) {
void* ptr = gbinder_writer_malloc(self, size);
if (ptr) {
memcpy(ptr, buf, size);
return ptr;
}
}
return NULL;
}
void
gbinder_writer_add_cleanup(
GBinderWriter* self,
GDestroyNotify destroy,
gpointer ptr) /* since 1.0.19 */
{
if (G_LIKELY(destroy)) {
GBinderWriterData* data = gbinder_writer_data(self);
if (G_LIKELY(data)) {
data->cleanup = gbinder_cleanup_add(data->cleanup, destroy, ptr);
}
}
}
/*
* Local Variables:
* mode: C

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Contact: Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -13,9 +13,9 @@
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of Jolla Ltd nor the names of its contributors may
* be used to endorse or promote products derived from this software
* without specific prior written permission.
* 3. Neither the names of the copyright holders nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
@@ -194,7 +194,7 @@ app_init(
{ "quiet", 'q', G_OPTION_FLAG_NO_ARG, G_OPTION_ARG_CALLBACK,
app_log_quiet, "Be quiet", NULL },
{ "async", 'a', 0, G_OPTION_ARG_NONE, &opt->async,
"Parform operations asynchronously", NULL },
"Perform operations asynchronously", NULL },
{ "device", 'd', 0, G_OPTION_ARG_STRING, &opt->dev,
"Binder device [" DEV_DEFAULT "]", "DEVICE" },
{ NULL }
@@ -245,7 +245,7 @@ int main(int argc, char* argv[])
app.opt = &opt;
if (app_init(&opt, argc, argv)) {
app.sm = gbinder_servicemanager_new(opt.dev);
if (app.sm) {
if (gbinder_servicemanager_wait(app.sm, -1)) {
if (opt.async) {
app_async(&app);
} else {

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -52,6 +52,7 @@ typedef struct app_options {
char* dev;
char* iface;
const char* name;
gboolean async;
} AppOptions;
typedef struct app {
@@ -62,6 +63,11 @@ typedef struct app {
int ret;
} App;
typedef struct response {
GBinderRemoteRequest* req;
GBinderLocalReply* reply;
} Response;
static const char pname[] = "binder-service";
static
@@ -76,6 +82,29 @@ app_signal(
return G_SOURCE_CONTINUE;
}
static
gboolean
app_async_resp(
gpointer user_data)
{
Response* resp = user_data;
gbinder_remote_request_complete(resp->req, resp->reply, 0);
return G_SOURCE_REMOVE;
}
static
void
app_async_free(
gpointer user_data)
{
Response* resp = user_data;
gbinder_local_reply_unref(resp->reply);
gbinder_remote_request_unref(resp->req);
g_free(resp);
}
static
GBinderLocalReply*
app_reply(
@@ -91,18 +120,30 @@ app_reply(
gbinder_remote_request_init_reader(req, &reader);
if (code == GBINDER_FIRST_CALL_TRANSACTION) {
const AppOptions* opt = app->opt;
const char* iface = gbinder_remote_request_interface(req);
if (!g_strcmp0(iface, app->opt->iface)) {
if (!g_strcmp0(iface, opt->iface)) {
char* str = gbinder_reader_read_string16(&reader);
GBinderLocalReply* reply = gbinder_local_object_new_reply(obj);
GVERBOSE("\"%s\" %u", iface, code);
GDEBUG("\"%s\"", str);
*status = 0;
gbinder_local_reply_append_string16(reply, str);
g_free(str);
*status = 0;
return reply;
if (opt->async) {
Response* resp = g_new0(Response, 1);
resp->reply = reply;
resp->req = gbinder_remote_request_ref(req);
g_idle_add_full(G_PRIORITY_DEFAULT_IDLE, app_async_resp,
resp, app_async_free);
gbinder_remote_request_block(resp->req);
return NULL;
} else {
return reply;
}
} else {
GDEBUG("Unexpected interface \"%s\"", iface);
}
@@ -140,6 +181,23 @@ app_add_service_done(
}
}
static
void
app_sm_presence_handler(
GBinderServiceManager* sm,
void* user_data)
{
App* app = user_data;
if (gbinder_servicemanager_is_present(app->sm)) {
GINFO("Service manager has reappeared");
gbinder_servicemanager_add_service(app->sm, app->opt->name, app->obj,
app_add_service_done, app);
} else {
GINFO("Service manager has died");
}
}
static
void
app_run(
@@ -148,6 +206,8 @@ app_run(
const char* name = app->opt->name;
guint sigtrm = g_unix_signal_add(SIGTERM, app_signal, app);
guint sigint = g_unix_signal_add(SIGINT, app_signal, app);
gulong presence_id = gbinder_servicemanager_add_presence_handler
(app->sm, app_sm_presence_handler, app);
app->loop = g_main_loop_new(NULL, TRUE);
@@ -158,6 +218,7 @@ app_run(
if (sigtrm) g_source_remove(sigtrm);
if (sigint) g_source_remove(sigint);
gbinder_servicemanager_remove_handler(app->sm, presence_id);
g_main_loop_unref(app->loop);
app->loop = NULL;
}
@@ -203,6 +264,8 @@ app_init(
"Binder device [" DEFAULT_DEVICE "]", "DEVICE" },
{ "interface", 'i', 0, G_OPTION_ARG_STRING, &opt->iface,
"Local interface [" DEFAULT_IFACE "]", "IFACE" },
{ "async", 'a', 0, G_OPTION_ARG_NONE, &opt->async,
"Handle calls asynchronously", NULL },
{ NULL }
};
@@ -254,7 +317,7 @@ int main(int argc, char* argv[])
app.opt = &opt;
if (app_init(&opt, argc, argv)) {
app.sm = gbinder_servicemanager_new(opt.dev);
if (app.sm) {
if (gbinder_servicemanager_wait(app.sm, -1)) {
app.obj = gbinder_servicemanager_new_local_object
(app.sm, opt.iface, app_reply, &app);
app_run(&app);

View File

@@ -16,6 +16,7 @@ all:
@$(MAKE) -C unit_remote_reply $*
@$(MAKE) -C unit_remote_request $*
@$(MAKE) -C unit_servicemanager $*
@$(MAKE) -C unit_servicename $*
@$(MAKE) -C unit_servicepoll $*
@$(MAKE) -C unit_writer $*

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -45,6 +45,7 @@ GLOG_MODULE_DEFINE2("test_binder", gutil_log_default);
static GHashTable* test_fd_map = NULL;
static GHashTable* test_node_map = NULL;
static GPrivate test_looper;
#define public_fd fd[0]
#define private_fd fd[1]
@@ -58,6 +59,22 @@ static GHashTable* test_node_map = NULL;
#define TF_ACCEPT_FDS 0x10
typedef struct test_binder_io TestBinderIo;
typedef struct test_binder TestBinder;
typedef
void
(*TestBinderPushDataFunc)(
int fd,
const void* data);
typedef struct test_binder_submit_thread {
GThread* thread;
GCond cond;
GMutex mutex;
gboolean run;
GSList* queue;
TestBinder* binder;
} TestBinderSubmitThread;
typedef struct test_binder_node {
char* path;
@@ -68,6 +85,8 @@ typedef struct test_binder_node {
typedef struct test_binder {
TestBinderNode* node;
TestBinderSubmitThread* submit_thread;
gboolean looper_enabled;
int fd[2];
} TestBinder;
@@ -104,6 +123,11 @@ typedef struct binder_pre_cookie_64 {
guint64 cookie;
} BinderPtrCookie64;
typedef struct binder_handle_cookie_64 {
guint32 handle;
guint64 cookie;
} __attribute__((packed)) BinderHandleCookie64;
#define BC_TRANSACTION_64 _IOW('c', 0, BinderTransactionData64)
#define BC_REPLY_64 _IOW('c', 1, BinderTransactionData64)
#define BC_FREE_BUFFER_64 _IOW('c', 3, guint64)
@@ -113,6 +137,8 @@ typedef struct binder_pre_cookie_64 {
#define BC_DECREFS _IOW('c', 7, guint32)
#define BC_ENTER_LOOPER _IO('c', 12)
#define BC_EXIT_LOOPER _IO('c', 13)
#define BC_REQUEST_DEATH_NOTIFICATION_64 _IOW('c', 14, BinderHandleCookie64)
#define BC_CLEAR_DEATH_NOTIFICATION_64 _IOW('c', 15, BinderHandleCookie64)
#define BR_TRANSACTION_64 _IOR('r', 2, BinderTransactionData64)
#define BR_REPLY_64 _IOR('r', 3, BinderTransactionData64)
@@ -126,6 +152,110 @@ typedef struct binder_pre_cookie_64 {
#define BR_DEAD_BINDER_64 _IOR('r', 15, guint64)
#define BR_FAILED_REPLY _IO('r', 17)
static
gpointer
test_binder_submit_thread_proc(
gpointer data)
{
TestBinderSubmitThread* submit = data;
TestBinder* binder = submit->binder;
GMutex* mutex = &submit->mutex;
GCond* cond = &submit->cond;
GDEBUG("Submit thread started");
g_mutex_lock(mutex);
while (submit->run) {
GBytes* next = NULL;
while (submit->run && !next) {
if (submit->queue) {
next = submit->queue->data;
submit->queue = g_slist_remove(submit->queue, next);
break;
} else {
g_cond_wait(cond, mutex);
}
}
if (next) {
int bytes_available = 0;
int err = ioctl(binder->public_fd, FIONREAD, &bytes_available);
/* Wait until the queue is empty */
g_assert(err >= 0);
while (bytes_available > 0 && submit->run) {
/* Wait a bit between polls */
g_cond_wait_until(cond, mutex, g_get_monotonic_time () +
100 * G_TIME_SPAN_MILLISECOND);
err = ioctl(binder->public_fd, FIONREAD, &bytes_available);
g_assert(err >= 0);
}
if (submit->run) {
gsize len;
gconstpointer data = g_bytes_get_data(next, &len);
GDEBUG("Submitting command 0x%08x", *(guint32*)data);
g_assert(write(binder->private_fd, data, len) == len);
}
g_bytes_unref(next);
}
}
g_mutex_unlock(mutex);
GDEBUG("Submit thread exiting");
return NULL;
}
static
TestBinderSubmitThread*
test_binder_submit_thread_new(
TestBinder* binder)
{
TestBinderSubmitThread* submit = g_new0(TestBinderSubmitThread, 1);
submit->run = TRUE;
submit->binder = binder;
g_cond_init(&submit->cond);
g_mutex_init(&submit->mutex);
submit->thread = g_thread_new(binder->node->path,
test_binder_submit_thread_proc, submit);
return submit;
}
static
void
test_binder_submit_thread_free(
TestBinderSubmitThread* submit)
{
if (submit) {
g_mutex_lock(&submit->mutex);
submit->run = FALSE;
g_cond_signal(&submit->cond);
g_mutex_unlock(&submit->mutex);
g_thread_join(submit->thread);
g_slist_free_full(submit->queue, (GDestroyNotify) g_bytes_unref);
g_cond_clear(&submit->cond);
g_mutex_clear(&submit->mutex);
g_free(submit);
}
}
static
void
test_binder_submit_later(
TestBinderSubmitThread* submit,
const void* data)
{
const guint32* cmd = data;
g_mutex_lock(&submit->mutex);
submit->queue = g_slist_append(submit->queue,
g_bytes_new(cmd, sizeof(*cmd) + _IOC_SIZE(*cmd)));
g_cond_signal(&submit->cond);
g_mutex_unlock(&submit->mutex);
}
static
void
test_io_free_buffer(
@@ -151,11 +281,11 @@ test_io_handle_write_read_64(
TestBinder* binder,
void* data)
{
int err, bytes_available = 0;
BinderWriteRead64* wr = data;
gssize bytes_left = wr->write_size - wr->write_consumed;
const guint8* write_ptr = (void*)(gsize)
(wr->write_buffer + wr->write_consumed);
gboolean is_looper;
while (bytes_left >= sizeof(guint32)) {
const guint cmd = *(guint32*)write_ptr;
@@ -176,12 +306,18 @@ test_io_handle_write_read_64(
test_io_free_buffer(binder,
GSIZE_TO_POINTER(*(guint64*)write_ptr));
break;
case BC_ENTER_LOOPER:
g_private_set(&test_looper, GINT_TO_POINTER(TRUE));
break;
case BC_EXIT_LOOPER:
g_private_set(&test_looper, NULL);
break;
case BC_REQUEST_DEATH_NOTIFICATION_64:
case BC_CLEAR_DEATH_NOTIFICATION_64:
case BC_INCREFS:
case BC_ACQUIRE:
case BC_RELEASE:
case BC_DECREFS:
case BC_ENTER_LOOPER:
case BC_EXIT_LOOPER:
break;
default:
#pragma message("TODO: implement more BINDER_WRITE_READ commands")
@@ -198,24 +334,45 @@ test_io_handle_write_read_64(
}
}
/* Now read the data from the socket */
err = ioctl(binder->public_fd, FIONREAD, &bytes_available);
if (err >= 0) {
int bytes_read = 0;
if (bytes_available > 0) {
bytes_read = read(binder->public_fd,
(void*)(gsize)(wr->read_buffer + wr->read_consumed),
wr->read_size - wr->read_consumed);
}
is_looper = g_private_get(&test_looper) ? TRUE : FALSE;
if (binder->looper_enabled || !is_looper) {
/* Now read the data from the socket */
int bytes_available = 0;
int err = ioctl(binder->public_fd, FIONREAD, &bytes_available);
if (bytes_read >= 0) {
wr->read_consumed += bytes_read;
return 0;
} else {
err = bytes_read;
if (err >= 0) {
int bytes_read = 0;
if (bytes_available >= 4) {
bytes_read = read(binder->public_fd,
(void*)(gsize)(wr->read_buffer + wr->read_consumed),
wr->read_size - wr->read_consumed);
} else {
struct timespec wait;
wait.tv_sec = 0;
wait.tv_nsec = 10 * 1000000; /* 10 ms */
nanosleep(&wait, &wait);
}
if (bytes_read >= 0) {
wr->read_consumed += bytes_read;
return 0;
} else {
err = bytes_read;
}
}
return err;
} else {
if (wr->read_size > 0) {
struct timespec wait;
wait.tv_sec = 0;
wait.tv_nsec = 100 * 1000000; /* 100 ms */
nanosleep(&wait, &wait);
}
return 0;
}
return err;
}
static const TestBinderIo test_io_64 = {
@@ -274,6 +431,17 @@ test_io_destroy_none(
GDEBUG("Not freeing %p", data);
}
void
test_binder_set_looper_enabled(
int fd,
gboolean enabled)
{
TestBinder* binder = test_binder_from_fd(fd);
g_assert(binder);
binder->looper_enabled = enabled;
}
void
test_binder_set_destroy(
int fd,
@@ -291,43 +459,35 @@ test_binder_set_destroy(
}
static
gboolean
void
test_binder_push_data(
int fd,
const void* data)
{
const guint32* cmd = data;
const int len = sizeof(*cmd) + _IOC_SIZE(*cmd);
TestBinder* binder = test_binder_from_fd(fd);
g_assert(binder);
g_assert(write(binder->private_fd, data, len) == len);
}
static
void
test_binder_push_data_later(
int fd,
const void* data)
{
TestBinder* binder = test_binder_from_fd(fd);
if (binder) {
const guint32* cmd = data;
const int len = sizeof(*cmd) + _IOC_SIZE(*cmd);
return write(binder->private_fd, data, len) == len;
g_assert(binder);
if (!binder->submit_thread) {
binder->submit_thread = test_binder_submit_thread_new(binder);
}
return FALSE;
test_binder_submit_later(binder->submit_thread, data);
}
static
void
test_binder_fill_transaction_data(
BinderTransactionData64* tr,
guint64 handle,
guint32 code,
const GByteArray* bytes)
{
g_assert(bytes);
memset(tr, 0, sizeof(*tr));
tr->handle = handle;
tr->code = code;
tr->data_size = bytes->len;
tr->sender_pid = getpid();
tr->sender_euid = geteuid();
/* This memory should eventually get deallocated with BC_FREE_BUFFER_64 */
tr->data_buffer = (gsize)g_memdup(bytes->data, bytes->len);
}
gboolean
test_binder_push_ptr_cookie(
int fd,
guint32 cmd,
@@ -339,60 +499,69 @@ test_binder_push_ptr_cookie(
memcpy(buf, &cmd, sizeof(cmd));
memset(data, 0, sizeof(*data));
data->ptr = (gsize)ptr;
return test_binder_push_data(fd, buf);
test_binder_push_data(fd, buf);
}
gboolean
void
test_binder_br_noop(
int fd)
{
guint32 cmd = BR_NOOP;
return test_binder_push_data(fd, &cmd);
test_binder_push_data(fd, &cmd);
}
gboolean
void
test_binder_br_increfs(
int fd,
void* ptr)
{
return test_binder_push_ptr_cookie(fd, BR_INCREFS_64, ptr);
test_binder_push_ptr_cookie(fd, BR_INCREFS_64, ptr);
}
gboolean
void
test_binder_br_acquire(
int fd,
void* ptr)
{
return test_binder_push_ptr_cookie(fd, BR_ACQUIRE_64, ptr);
test_binder_push_ptr_cookie(fd, BR_ACQUIRE_64, ptr);
}
gboolean
void
test_binder_br_release(
int fd,
void* ptr)
{
return test_binder_push_ptr_cookie(fd, BR_RELEASE_64, ptr);
test_binder_push_ptr_cookie(fd, BR_RELEASE_64, ptr);
}
gboolean
void
test_binder_br_decrefs(
int fd,
void* ptr)
{
return test_binder_push_ptr_cookie(fd, BR_DECREFS_64, ptr);
test_binder_push_ptr_cookie(fd, BR_DECREFS_64, ptr);
}
gboolean
void
test_binder_br_transaction_complete(
int fd)
{
guint32 cmd = BR_TRANSACTION_COMPLETE;
return test_binder_push_data(fd, &cmd);
test_binder_push_data(fd, &cmd);
}
gboolean
void
test_binder_br_transaction_complete_later(
int fd)
{
guint32 cmd = BR_TRANSACTION_COMPLETE;
test_binder_push_data_later(fd, &cmd);
}
void
test_binder_br_dead_binder(
int fd,
guint handle)
@@ -403,28 +572,47 @@ test_binder_br_dead_binder(
buf[0] = BR_DEAD_BINDER_64;
memcpy(buf + 1, &handle64, sizeof(handle64));
return test_binder_push_data(fd, buf);
test_binder_push_data(fd, buf);
}
gboolean
void
test_binder_br_dead_reply(
int fd)
{
guint32 cmd = BR_DEAD_REPLY;
return test_binder_push_data(fd, &cmd);
test_binder_push_data(fd, &cmd);
}
gboolean
void
test_binder_br_failed_reply(
int fd)
{
guint32 cmd = BR_FAILED_REPLY;
return test_binder_push_data(fd, &cmd);
test_binder_push_data(fd, &cmd);
}
gboolean
static
void
test_binder_fill_transaction_data(
BinderTransactionData64* tr,
guint64 handle,
guint32 code,
const GByteArray* bytes)
{
memset(tr, 0, sizeof(*tr));
tr->handle = handle;
tr->code = code;
tr->data_size = bytes ? bytes->len : 0;
tr->sender_pid = getpid();
tr->sender_euid = geteuid();
/* This memory should eventually get deallocated with BC_FREE_BUFFER_64 */
tr->data_buffer = (gsize)g_memdup(bytes ? (void*)bytes->data : (void*)"",
tr->data_size);
}
void
test_binder_br_transaction(
int fd,
void* target,
@@ -438,15 +626,17 @@ test_binder_br_transaction(
test_binder_fill_transaction_data((void*)(buf + sizeof(cmd)),
(gsize)target, code, bytes);
return test_binder_push_data(fd, buf);
test_binder_push_data(fd, buf);
}
gboolean
test_binder_br_reply(
static
void
test_binder_br_reply1(
int fd,
guint32 handle,
guint32 code,
const GByteArray* bytes)
const GByteArray* bytes,
TestBinderPushDataFunc push)
{
guint32 cmd = BR_REPLY_64;
guint8 buf[sizeof(guint32) + sizeof(BinderTransactionData64)];
@@ -455,13 +645,34 @@ test_binder_br_reply(
test_binder_fill_transaction_data((void*)(buf + sizeof(cmd)),
handle, code, bytes);
return test_binder_push_data(fd, buf);
push(fd, buf);
}
gboolean
test_binder_br_reply_status(
void
test_binder_br_reply(
int fd,
gint32 status)
guint32 handle,
guint32 code,
const GByteArray* bytes)
{
test_binder_br_reply1(fd, handle, code, bytes, test_binder_push_data);
}
void
test_binder_br_reply_later(
int fd,
guint32 handle,
guint32 code,
const GByteArray* bytes)
{
test_binder_br_reply1(fd, handle, code, bytes, test_binder_push_data_later);
}
void
test_binder_br_reply_status1(
int fd,
gint32 status,
TestBinderPushDataFunc push)
{
guint8 buf[sizeof(guint32) + sizeof(BinderTransactionData64)];
guint32* cmd = (void*)buf;
@@ -473,7 +684,24 @@ test_binder_br_reply_status(
tr->data_size = sizeof(status);
/* This memory should eventually get deallocated with BC_FREE_BUFFER_64 */
tr->data_buffer = (gsize)g_memdup(&status, sizeof(status));
return test_binder_push_data(fd, buf);
push(fd, buf);
}
void
test_binder_br_reply_status(
int fd,
gint32 status)
{
test_binder_br_reply_status1(fd, status, test_binder_push_data);
}
void
test_binder_br_reply_status_later(
int fd,
gint32 status)
{
test_binder_br_reply_status1(fd, status, test_binder_push_data_later);
}
int
@@ -532,6 +760,7 @@ gbinder_system_close(
g_hash_table_unref(test_fd_map);
test_fd_map = NULL;
}
test_binder_submit_thread_free(binder->submit_thread);
test_binder_node_unref(binder->node);
close(binder->public_fd);
close(binder->private_fd);

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -35,66 +35,87 @@
#include "test_common.h"
gboolean
void
test_binder_br_noop(
int fd);
gboolean
void
test_binder_br_increfs(
int fd,
void* ptr);
gboolean
void
test_binder_br_acquire(
int fd,
void* ptr);
gboolean
void
test_binder_br_release(
int fd,
void* ptr);
gboolean
void
test_binder_br_decrefs(
int fd,
void* ptr);
gboolean
void
test_binder_br_transaction_complete(
int fd);
gboolean
void
test_binder_br_transaction_complete_later(
int fd);
void
test_binder_br_dead_binder(
int fd,
guint handle);
gboolean
void
test_binder_br_dead_reply(
int fd);
gboolean
void
test_binder_br_failed_reply(
int fd);
gboolean
void
test_binder_br_transaction(
int fd,
void* target,
guint32 code,
const GByteArray* bytes);
gboolean
void
test_binder_br_reply(
int fd,
guint32 handle,
guint32 code,
const GByteArray* bytes);
gboolean
void
test_binder_br_reply_status(
int fd,
gint32 status);
void
test_binder_br_reply_later(
int fd,
guint32 handle,
guint32 code,
const GByteArray* bytes);
void
test_binder_br_reply_status_later(
int fd,
gint32 status);
void
test_binder_set_looper_enabled(
int fd,
gboolean enabled);
void
test_binder_set_destroy(
int fd,

View File

@@ -18,6 +18,7 @@ unit_remote_object \
unit_remote_reply \
unit_remote_request \
unit_servicemanager \
unit_servicename \
unit_servicepoll \
unit_writer"

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -74,16 +74,15 @@ void
test_null(
void)
{
GBinderClient* null = NULL;
g_assert(!gbinder_client_new(NULL, NULL));
g_assert(!gbinder_client_ref(null));
gbinder_client_unref(null);
g_assert(!gbinder_client_ref(NULL));
g_assert(!gbinder_client_interface(NULL));
gbinder_client_unref(NULL);
g_assert(!gbinder_client_new_request(NULL));
g_assert(!gbinder_client_transact_sync_reply(null, 0, NULL, NULL));
g_assert(gbinder_client_transact_sync_oneway(null, 0, NULL) == (-EINVAL));
g_assert(!gbinder_client_transact(null, 0, 0, NULL, NULL, NULL, NULL));
gbinder_client_cancel(null, 0);
g_assert(!gbinder_client_transact_sync_reply(NULL, 0, NULL, NULL));
g_assert(gbinder_client_transact_sync_oneway(NULL, 0, NULL) == (-EINVAL));
g_assert(!gbinder_client_transact(NULL, 0, 0, NULL, NULL, NULL, NULL));
gbinder_client_cancel(NULL, 0);
}
/*==========================================================================*
@@ -98,10 +97,12 @@ test_basic(
GBinderIpc* ipc = gbinder_ipc_new(GBINDER_DEFAULT_BINDER, NULL);
GBinderObjectRegistry* reg = gbinder_ipc_object_registry(ipc);
GBinderRemoteObject* obj = gbinder_object_registry_get_remote(reg, 0);
GBinderClient* client = gbinder_client_new(obj, "foo");
const char* iface = "foo";
GBinderClient* client = gbinder_client_new(obj, iface);
g_assert(client);
g_assert(gbinder_client_ref(client) == client);
g_assert(!g_strcmp0(gbinder_client_interface(client), iface));
gbinder_client_unref(client);
gbinder_client_cancel(client, 0); /* does nothing */
@@ -182,10 +183,10 @@ test_sync_reply_tx(
data = gbinder_local_reply_data(reply);
g_assert(data);
g_assert(test_binder_br_noop(fd));
g_assert(test_binder_br_transaction_complete(fd));
g_assert(test_binder_br_noop(fd));
g_assert(test_binder_br_reply(fd, handle, code, data->bytes));
test_binder_br_noop(fd);
test_binder_br_transaction_complete(fd);
test_binder_br_noop(fd);
test_binder_br_reply(fd, handle, code, data->bytes);
tx_reply = gbinder_client_transact_sync_reply(client, 0, req, &status);
g_assert(tx_reply);
@@ -283,10 +284,10 @@ test_reply_tx(
data = gbinder_local_reply_data(reply);
g_assert(data);
g_assert(test_binder_br_noop(fd));
g_assert(test_binder_br_transaction_complete(fd));
g_assert(test_binder_br_noop(fd));
g_assert(test_binder_br_reply(fd, handle, code, data->bytes));
test_binder_br_noop(fd);
test_binder_br_transaction_complete(fd);
test_binder_br_noop(fd);
test_binder_br_reply(fd, handle, code, data->bytes);
id = gbinder_client_transact(client, 0, 0, req, done, destroy, loop);
g_assert(id);

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -91,7 +91,7 @@ test_noop(
g_assert(driver);
g_assert(fd >= 0);
g_assert(test_binder_br_noop(fd));
test_binder_br_noop(fd);
g_assert(gbinder_driver_poll(driver, NULL) == POLLIN);
g_assert(gbinder_driver_read(driver, NULL, NULL) == 0);

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -52,6 +52,24 @@
static TestOpt test_opt;
static
gboolean
test_unref_ipc(
gpointer ipc)
{
gbinder_ipc_unref(ipc);
return G_SOURCE_REMOVE;
}
static
void
test_quit_when_destroyed(
gpointer loop,
GObject* obj)
{
test_quit_later((GMainLoop*)loop);
}
/*==========================================================================*
* null
*==========================================================================*/
@@ -111,6 +129,45 @@ test_basic(
g_assert(!gbinder_ipc_new("invalid path", NULL));
}
/*==========================================================================*
* async_oneway
*==========================================================================*/
static
void
test_async_oneway_done(
GBinderIpc* ipc,
GBinderRemoteReply* reply,
int status,
void* user_data)
{
g_assert(!status);
g_assert(!reply);
test_quit_later((GMainLoop*)user_data);
}
static
void
test_async_oneway(
void)
{
GBinderIpc* ipc = gbinder_ipc_new(GBINDER_DEFAULT_BINDER, NULL);
const GBinderIo* io = gbinder_driver_io(ipc->driver);
const int fd = gbinder_driver_fd(ipc->driver);
GBinderLocalRequest* req = gbinder_local_request_new(io, NULL);
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
gulong id;
test_binder_br_transaction_complete(fd);
id = gbinder_ipc_transact(ipc, 0, 1, GBINDER_TX_FLAG_ONEWAY,
req, test_async_oneway_done, NULL, loop);
g_assert(id);
test_run(&test_opt, loop);
gbinder_local_request_unref(req);
gbinder_ipc_unref(ipc);
}
/*==========================================================================*
* sync_oneway
*==========================================================================*/
@@ -125,7 +182,7 @@ test_sync_oneway(
const int fd = gbinder_driver_fd(ipc->driver);
GBinderLocalRequest* req = gbinder_local_request_new(io, NULL);
g_assert(test_binder_br_transaction_complete(fd));
test_binder_br_transaction_complete(fd);
g_assert(gbinder_ipc_transact_sync_oneway(ipc, 0, 1, req) ==
GBINDER_STATUS_OK);
gbinder_local_request_unref(req);
@@ -157,10 +214,10 @@ test_sync_reply_ok_status(
data = gbinder_local_reply_data(reply);
g_assert(data);
g_assert(test_binder_br_noop(fd));
g_assert(test_binder_br_transaction_complete(fd));
g_assert(test_binder_br_noop(fd));
g_assert(test_binder_br_reply(fd, handle, code, data->bytes));
test_binder_br_noop(fd);
test_binder_br_transaction_complete(fd);
test_binder_br_noop(fd);
test_binder_br_reply(fd, handle, code, data->bytes);
tx_reply = gbinder_ipc_transact_sync_reply(ipc, handle, code, req, status);
g_assert(tx_reply);
@@ -205,10 +262,10 @@ test_sync_reply_error(
const gint expected_status = GBINDER_STATUS_FAILED;
int status = INT_MAX;
g_assert(test_binder_br_noop(fd));
g_assert(test_binder_br_transaction_complete(fd));
g_assert(test_binder_br_noop(fd));
g_assert(test_binder_br_reply_status(fd, expected_status));
test_binder_br_noop(fd);
test_binder_br_transaction_complete(fd);
test_binder_br_noop(fd);
test_binder_br_reply_status(fd, expected_status);
g_assert(!gbinder_ipc_transact_sync_reply(ipc, handle, code, req, &status));
g_assert(status == expected_status);
@@ -268,10 +325,10 @@ test_transact_ok(
data = gbinder_local_reply_data(reply);
g_assert(data);
g_assert(test_binder_br_noop(fd));
g_assert(test_binder_br_transaction_complete(fd));
g_assert(test_binder_br_noop(fd));
g_assert(test_binder_br_reply(fd, handle, code, data->bytes));
test_binder_br_noop(fd);
test_binder_br_transaction_complete(fd);
test_binder_br_noop(fd);
test_binder_br_reply(fd, handle, code, data->bytes);
id = gbinder_ipc_transact(ipc, handle, code, 0, req,
test_transact_ok_done, test_transact_ok_destroy, loop);
@@ -317,8 +374,8 @@ test_transact_dead(
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
gulong id;
g_assert(test_binder_br_noop(fd));
g_assert(test_binder_br_dead_reply(fd));
test_binder_br_noop(fd);
test_binder_br_dead_reply(fd);
id = gbinder_ipc_transact(ipc, 1, 2, 0, req, test_transact_dead_done,
NULL, loop);
@@ -363,8 +420,8 @@ test_transact_failed(
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
gulong id;
g_assert(test_binder_br_noop(fd));
g_assert(test_binder_br_failed_reply(fd));
test_binder_br_noop(fd);
test_binder_br_failed_reply(fd);
id = gbinder_ipc_transact(ipc, 1, 2, 0, req, test_transact_failed_done,
NULL, loop);
@@ -411,8 +468,8 @@ test_transact_status(
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
gulong id;
g_assert(test_binder_br_noop(fd));
g_assert(test_binder_br_reply_status(fd, EXPECTED_STATUS));
test_binder_br_noop(fd);
test_binder_br_reply_status(fd, EXPECTED_STATUS);
id = gbinder_ipc_transact(ipc, 1, 2, 0, req, test_transact_status_done,
NULL, loop);
@@ -606,24 +663,6 @@ test_transact_incoming_proc(
return gbinder_local_object_new_reply(obj);
}
static
gboolean
test_transact_unref_ipc(
gpointer ipc)
{
gbinder_ipc_unref(ipc);
return G_SOURCE_REMOVE;
}
static
void
test_transact_done(
gpointer loop,
GObject* ipc)
{
test_quit_later((GMainLoop*)loop);
}
static
void
test_transact_incoming(
@@ -647,21 +686,22 @@ test_transact_incoming(
data = gbinder_local_request_data(req);
test_binder_br_transaction(fd, obj, 1, data->bytes);
test_binder_set_looper_enabled(fd, TRUE);
test_run(&test_opt, loop);
/* Now we need to wait until GBinderIpc is destroyed */
GDEBUG("waiting for GBinderIpc to get destroyed");
g_object_weak_ref(G_OBJECT(ipc), test_transact_done, loop);
g_object_weak_ref(G_OBJECT(ipc), test_quit_when_destroyed, loop);
gbinder_local_object_unref(obj);
gbinder_local_request_unref(req);
g_idle_add(test_transact_unref_ipc, ipc);
g_idle_add(test_unref_ipc, ipc);
test_run(&test_opt, loop);
g_main_loop_unref(loop);
}
/*==========================================================================*
* transact_incoming_status
* transact_status_reply
*==========================================================================*/
static
@@ -708,14 +748,191 @@ test_transact_status_reply(
data = gbinder_local_request_data(req);
test_binder_br_transaction(fd, obj, 1, data->bytes);
test_binder_set_looper_enabled(fd, TRUE);
test_run(&test_opt, loop);
/* Now we need to wait until GBinderIpc is destroyed */
GDEBUG("waiting for GBinderIpc to get destroyed");
g_object_weak_ref(G_OBJECT(ipc), test_transact_done, loop);
g_object_weak_ref(G_OBJECT(ipc), test_quit_when_destroyed, loop);
gbinder_local_object_unref(obj);
gbinder_local_request_unref(req);
g_idle_add(test_transact_unref_ipc, ipc);
g_idle_add(test_unref_ipc, ipc);
test_run(&test_opt, loop);
g_main_loop_unref(loop);
}
/*==========================================================================*
* transact_async
*==========================================================================*/
typedef struct test_transact_async_req {
GBinderLocalObject* obj;
GBinderRemoteRequest* req;
GMainLoop* loop;
} TestTransactAsyncReq;
static
void
test_transact_async_done(
gpointer data)
{
TestTransactAsyncReq* test = data;
gbinder_local_object_unref(test->obj);
gbinder_remote_request_unref(test->req);
test_quit_later(test->loop);
g_free(test);
}
static
gboolean
test_transact_async_reply(
gpointer data)
{
TestTransactAsyncReq* test = data;
GBinderLocalReply* reply = gbinder_local_object_new_reply(test->obj);
gbinder_remote_request_complete(test->req, reply, 0);
gbinder_local_reply_unref(reply);
return G_SOURCE_REMOVE;
}
static
GBinderLocalReply*
test_transact_async_proc(
GBinderLocalObject* obj,
GBinderRemoteRequest* req,
guint code,
guint flags,
int* status,
void* loop)
{
TestTransactAsyncReq* test = g_new(TestTransactAsyncReq, 1);
GVERBOSE_("\"%s\" %u", gbinder_remote_request_interface(req), code);
g_assert(!flags);
g_assert(gbinder_remote_request_sender_pid(req) == getpid());
g_assert(gbinder_remote_request_sender_euid(req) == geteuid());
g_assert(!g_strcmp0(gbinder_remote_request_interface(req), "test"));
g_assert(!g_strcmp0(gbinder_remote_request_read_string8(req), "message"));
g_assert(code == 1);
test->obj = gbinder_local_object_ref(obj);
test->req = gbinder_remote_request_ref(req);
test->loop = (GMainLoop*)loop;
gbinder_remote_request_block(req);
gbinder_remote_request_block(req); /* wrong state; has no effect */
g_idle_add_full(G_PRIORITY_DEFAULT_IDLE, test_transact_async_reply, test,
test_transact_async_done);
return NULL;
}
static
void
test_transact_async(
void)
{
GBinderIpc* ipc = gbinder_ipc_new(GBINDER_DEFAULT_BINDER, NULL);
const GBinderIo* io = gbinder_driver_io(ipc->driver);
const int fd = gbinder_driver_fd(ipc->driver);
const char* dev = gbinder_driver_dev(ipc->driver);
const GBinderRpcProtocol* prot = gbinder_rpc_protocol_for_device(dev);
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
GBinderLocalObject* obj = gbinder_ipc_new_local_object
(ipc, "test", test_transact_async_proc, loop);
GBinderLocalRequest* req = gbinder_local_request_new(io, NULL);
GBinderOutputData* data;
GBinderWriter writer;
gbinder_local_request_init_writer(req, &writer);
prot->write_rpc_header(&writer, "test");
gbinder_writer_append_string8(&writer, "message");
data = gbinder_local_request_data(req);
test_binder_br_transaction(fd, obj, 1, data->bytes);
test_binder_set_looper_enabled(fd, TRUE);
test_run(&test_opt, loop);
/* Now we need to wait until GBinderIpc is destroyed */
GDEBUG("waiting for GBinderIpc to get destroyed");
g_object_weak_ref(G_OBJECT(ipc), test_quit_when_destroyed, loop);
gbinder_local_object_unref(obj);
gbinder_local_request_unref(req);
g_idle_add(test_unref_ipc, ipc);
test_run(&test_opt, loop);
g_main_loop_unref(loop);
}
/*==========================================================================*
* transact_async_sync
*==========================================================================*/
static
GBinderLocalReply*
test_transact_async_sync_proc(
GBinderLocalObject* obj,
GBinderRemoteRequest* req,
guint code,
guint flags,
int* status,
void* loop)
{
GBinderLocalReply* reply = gbinder_local_object_new_reply(obj);
GVERBOSE_("\"%s\" %u", gbinder_remote_request_interface(req), code);
g_assert(!flags);
g_assert(gbinder_remote_request_sender_pid(req) == getpid());
g_assert(gbinder_remote_request_sender_euid(req) == geteuid());
g_assert(!g_strcmp0(gbinder_remote_request_interface(req), "test"));
g_assert(!g_strcmp0(gbinder_remote_request_read_string8(req), "message"));
g_assert(code == 1);
/* Block and immediately complete the call */
gbinder_remote_request_block(req);
gbinder_remote_request_complete(req, reply, 0);
gbinder_remote_request_complete(req, reply, 0); /* This one is ignored */
gbinder_local_reply_unref(reply);
test_quit_later((GMainLoop*)loop);
return NULL;
}
static
void
test_transact_async_sync(
void)
{
GBinderIpc* ipc = gbinder_ipc_new(GBINDER_DEFAULT_BINDER, NULL);
const GBinderIo* io = gbinder_driver_io(ipc->driver);
const int fd = gbinder_driver_fd(ipc->driver);
const char* dev = gbinder_driver_dev(ipc->driver);
const GBinderRpcProtocol* prot = gbinder_rpc_protocol_for_device(dev);
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
GBinderLocalObject* obj = gbinder_ipc_new_local_object
(ipc, "test", test_transact_async_sync_proc, loop);
GBinderLocalRequest* req = gbinder_local_request_new(io, NULL);
GBinderOutputData* data;
GBinderWriter writer;
gbinder_local_request_init_writer(req, &writer);
prot->write_rpc_header(&writer, "test");
gbinder_writer_append_string8(&writer, "message");
data = gbinder_local_request_data(req);
test_binder_br_transaction(fd, obj, 1, data->bytes);
test_binder_set_looper_enabled(fd, TRUE);
test_run(&test_opt, loop);
/* Now we need to wait until GBinderIpc is destroyed */
GDEBUG("waiting for GBinderIpc to get destroyed");
g_object_weak_ref(G_OBJECT(ipc), test_quit_when_destroyed, loop);
gbinder_local_object_unref(obj);
gbinder_local_request_unref(req);
g_idle_add(test_unref_ipc, ipc);
test_run(&test_opt, loop);
g_main_loop_unref(loop);
@@ -726,26 +943,29 @@ test_transact_status_reply(
*==========================================================================*/
#define TEST_PREFIX "/ipc/"
#define TEST_(t) TEST_PREFIX t
int main(int argc, char* argv[])
{
g_test_init(&argc, &argv, NULL);
g_test_add_func(TEST_PREFIX "null", test_null);
g_test_add_func(TEST_PREFIX "basic", test_basic);
g_test_add_func(TEST_PREFIX "sync_oneway", test_sync_oneway);
g_test_add_func(TEST_PREFIX "sync_reply_ok", test_sync_reply_ok);
g_test_add_func(TEST_PREFIX "sync_reply_error", test_sync_reply_error);
g_test_add_func(TEST_PREFIX "transact_ok", test_transact_ok);
g_test_add_func(TEST_PREFIX "transact_dead", test_transact_dead);
g_test_add_func(TEST_PREFIX "transact_failed", test_transact_failed);
g_test_add_func(TEST_PREFIX "transact_status", test_transact_status);
g_test_add_func(TEST_PREFIX "transact_custom", test_transact_custom);
g_test_add_func(TEST_PREFIX "transact_custom2", test_transact_custom2);
g_test_add_func(TEST_PREFIX "transact_cancel", test_transact_cancel);
g_test_add_func(TEST_PREFIX "transact_cancel2", test_transact_cancel2);
g_test_add_func(TEST_PREFIX "transact_incoming", test_transact_incoming);
g_test_add_func(TEST_PREFIX "transact_status_reply",
test_transact_status_reply);
g_test_add_func(TEST_("null"), test_null);
g_test_add_func(TEST_("basic"), test_basic);
g_test_add_func(TEST_("async_oneway"), test_async_oneway);
g_test_add_func(TEST_("sync_oneway"), test_sync_oneway);
g_test_add_func(TEST_("sync_reply_ok"), test_sync_reply_ok);
g_test_add_func(TEST_("sync_reply_error"), test_sync_reply_error);
g_test_add_func(TEST_("transact_ok"), test_transact_ok);
g_test_add_func(TEST_("transact_dead"), test_transact_dead);
g_test_add_func(TEST_("transact_failed"), test_transact_failed);
g_test_add_func(TEST_("transact_status"), test_transact_status);
g_test_add_func(TEST_("transact_custom"), test_transact_custom);
g_test_add_func(TEST_("transact_custom2"), test_transact_custom2);
g_test_add_func(TEST_("transact_cancel"), test_transact_cancel);
g_test_add_func(TEST_("transact_cancel2"), test_transact_cancel2);
g_test_add_func(TEST_("transact_incoming"), test_transact_incoming);
g_test_add_func(TEST_("transact_status_reply"), test_transact_status_reply);
g_test_add_func(TEST_("transact_async"), test_transact_async);
g_test_add_func(TEST_("transact_async_sync"), test_transact_async_sync);
test_init(&test_opt, argc, argv);
return g_test_run();
}

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -195,6 +195,101 @@ static
void
test_ping(
void)
{
int status = INT_MAX;
const char* dev = GBINDER_DEFAULT_HWBINDER;
const GBinderRpcProtocol* prot = gbinder_rpc_protocol_for_device(dev);
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
GBinderObjectRegistry* reg = gbinder_ipc_object_registry(ipc);
GBinderRemoteRequest* req = gbinder_remote_request_new(reg, prot, 0, 0);
GBinderLocalObject* obj =
gbinder_ipc_new_local_object(ipc, NULL, NULL, NULL);
GBinderLocalReply* reply;
GBinderOutputData* out_data;
static const guint8 result[] = { 0x00, 0x00, 0x00, 0x00 };
g_assert(gbinder_local_object_can_handle_transaction(obj, NULL,
GBINDER_PING_TRANSACTION) == GBINDER_LOCAL_TRANSACTION_LOOPER);
/* If can_handle_transaction() returns TRANSACTION_LOOPER then it must be
* handled by handle_looper_transaction() */
g_assert(!gbinder_local_object_handle_transaction(obj, req,
GBINDER_PING_TRANSACTION, 0, &status));
g_assert(status == (-EBADMSG));
reply = gbinder_local_object_handle_looper_transaction(obj, req,
GBINDER_PING_TRANSACTION, 0, &status);
g_assert(reply);
g_assert(status == GBINDER_STATUS_OK);
out_data = gbinder_local_reply_data(reply);
g_assert(out_data);
g_assert(out_data->bytes);
g_assert(out_data->bytes->len == sizeof(result));
g_assert(!memcmp(out_data->bytes->data, result, sizeof(result)));
gbinder_ipc_unref(ipc);
gbinder_local_object_unref(obj);
gbinder_local_reply_unref(reply);
gbinder_remote_request_unref(req);
}
/*==========================================================================*
* interface
*==========================================================================*/
static
void
test_interface(
void)
{
int status = INT_MAX;
const char* dev = GBINDER_DEFAULT_HWBINDER;
const GBinderRpcProtocol* prot = gbinder_rpc_protocol_for_device(dev);
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
GBinderObjectRegistry* reg = gbinder_ipc_object_registry(ipc);
GBinderRemoteRequest* req = gbinder_remote_request_new(reg, prot, 0, 0);
GBinderLocalObject* obj =
gbinder_ipc_new_local_object(ipc, "x", NULL, NULL);
GBinderLocalReply* reply;
GBinderOutputData* out_data;
static const guint8 result[] = {
TEST_INT32_BYTES(1),
TEST_INT16_BYTES('x'), 0x00, 0x00
};
g_assert(gbinder_local_object_can_handle_transaction(obj, NULL,
GBINDER_INTERFACE_TRANSACTION) == GBINDER_LOCAL_TRANSACTION_LOOPER);
/* If can_handle_transaction() returns TRANSACTION_LOOPER then it must be
* handled by handle_looper_transaction() */
g_assert(!gbinder_local_object_handle_transaction(obj, req,
GBINDER_INTERFACE_TRANSACTION, 0, &status));
g_assert(status == (-EBADMSG));
reply = gbinder_local_object_handle_looper_transaction(obj, req,
GBINDER_INTERFACE_TRANSACTION, 0, &status);
g_assert(reply);
g_assert(status == GBINDER_STATUS_OK);
out_data = gbinder_local_reply_data(reply);
g_assert(out_data);
g_assert(out_data->bytes);
g_assert(out_data->bytes->len == sizeof(result));
g_assert(!memcmp(out_data->bytes->data, result, sizeof(result)));
gbinder_ipc_unref(ipc);
gbinder_local_object_unref(obj);
gbinder_local_reply_unref(reply);
gbinder_remote_request_unref(req);
}
/*==========================================================================*
* hidl_ping
*==========================================================================*/
static
void
test_hidl_ping(
void)
{
static const guint8 req_data [] = { TEST_BASE_INTERFACE_HEADER_BYTES };
int status = INT_MAX;
@@ -206,6 +301,8 @@ test_ping(
GBinderLocalObject* obj =
gbinder_ipc_new_local_object(ipc, NULL, NULL, NULL);
GBinderLocalReply* reply;
GBinderOutputData* out_data;
static const guint8 result[] = { 0x00, 0x00, 0x00, 0x00 };
gbinder_remote_request_set_data(req, HIDL_PING_TRANSACTION,
gbinder_buffer_new(ipc->driver, g_memdup(req_data, sizeof(req_data)),
@@ -224,6 +321,12 @@ test_ping(
g_assert(reply);
g_assert(status == GBINDER_STATUS_OK);
out_data = gbinder_local_reply_data(reply);
g_assert(out_data);
g_assert(out_data->bytes);
g_assert(out_data->bytes->len == sizeof(result));
g_assert(!memcmp(out_data->bytes->data, result, sizeof(result)));
gbinder_ipc_unref(ipc);
gbinder_local_object_unref(obj);
gbinder_local_reply_unref(reply);
@@ -525,7 +628,7 @@ test_increfs(
/* ipc is not an object, will be ignored */
test_binder_br_increfs(fd, ipc);
test_binder_br_increfs(fd, obj);
test_binder_set_looper_enabled(fd, TRUE);
test_run(&test_opt, loop);
g_assert(obj->weak_refs == 1);
@@ -568,7 +671,7 @@ test_decrefs(
test_binder_br_decrefs(fd, ipc);
test_binder_br_increfs(fd, obj);
test_binder_br_decrefs(fd, obj);
test_binder_set_looper_enabled(fd, TRUE);
test_run(&test_opt, loop);
g_assert(obj->weak_refs == 0);
@@ -609,7 +712,7 @@ test_acquire(
/* ipc is not an object, will be ignored */
test_binder_br_acquire(fd, ipc);
test_binder_br_acquire(fd, obj);
test_binder_set_looper_enabled(fd, TRUE);
test_run(&test_opt, loop);
g_assert(obj->strong_refs == 1);
@@ -652,7 +755,7 @@ test_release(
test_binder_br_release(fd, ipc);
test_binder_br_acquire(fd, obj);
test_binder_br_release(fd, obj);
test_binder_set_looper_enabled(fd, TRUE);
test_run(&test_opt, loop);
g_assert(obj->strong_refs == 0);
@@ -674,6 +777,8 @@ int main(int argc, char* argv[])
g_test_add_func(TEST_PREFIX "null", test_null);
g_test_add_func(TEST_PREFIX "basic", test_basic);
g_test_add_func(TEST_PREFIX "ping", test_ping);
g_test_add_func(TEST_PREFIX "interface", test_interface);
g_test_add_func(TEST_PREFIX "hidl_ping", test_hidl_ping);
g_test_add_func(TEST_PREFIX "get_descriptor", test_get_descriptor);
g_test_add_func(TEST_PREFIX "descriptor_chain", test_descriptor_chain);
g_test_add_func(TEST_PREFIX "custom_iface", test_custom_iface);

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -100,6 +100,7 @@ test_empty(
g_assert(!gbinder_reader_read_hidl_vec1(&reader, &count, 1));
g_assert(!count);
g_assert(!elemsize);
g_assert(!gbinder_reader_skip_hidl_string(&reader));
g_assert(!gbinder_reader_read_hidl_string(&reader));
g_assert(!gbinder_reader_read_hidl_string_vec(&reader));
g_assert(!gbinder_reader_skip_buffer(&reader));
@@ -451,8 +452,8 @@ test_string16_null(
GBinderDriver* driver = gbinder_driver_new(GBINDER_DEFAULT_BINDER, NULL);
GBinderReader reader;
GBinderReaderData data;
gunichar2* out2 = NULL;
gsize len = 0;
const gunichar2* out2 = NULL;
gsize len = 1;
char dummy;
char* out = &dummy;
@@ -466,12 +467,23 @@ test_string16_null(
g_assert(gbinder_reader_read_nullable_string16_utf16(&reader, NULL, NULL));
g_assert(gbinder_reader_at_end(&reader));
len = 1;
gbinder_reader_init(&reader, &data, 0, sizeof(test_string16_in_null));
g_assert(gbinder_reader_read_nullable_string16_utf16(&reader, &out2, &len));
g_assert(gbinder_reader_at_end(&reader));
g_assert(!out2);
g_assert(!len);
gbinder_reader_init(&reader, &data, 0, sizeof(test_string16_in_null));
g_assert(!gbinder_reader_read_string16_utf16(&reader, NULL));
g_assert(gbinder_reader_at_end(&reader));
len = 1;
gbinder_reader_init(&reader, &data, 0, sizeof(test_string16_in_null));
g_assert(!gbinder_reader_read_string16_utf16(&reader, &len));
g_assert(gbinder_reader_at_end(&reader));
g_assert(!len);
gbinder_reader_init(&reader, &data, 0, sizeof(test_string16_in_null));
g_assert(gbinder_reader_read_nullable_string16(&reader, NULL));
g_assert(gbinder_reader_at_end(&reader));
@@ -503,7 +515,7 @@ test_string16(
GBinderReader reader;
GBinderReaderData data;
const gboolean valid = (test->out != NULL);
gunichar2* out2 = NULL;
const gunichar2* out2 = NULL;
gsize len = 0;
char* str = NULL;
@@ -512,6 +524,18 @@ test_string16(
data.buffer = gbinder_buffer_new(driver, g_memdup(test->in, test->in_size),
test->in_size, NULL);
gbinder_reader_init(&reader, &data, 0, test->in_size);
if (valid) {
out2 = gbinder_reader_read_string16_utf16(&reader, &len);
g_assert(out2);
g_assert((gsize)len == strlen(test->out));
g_assert(gbinder_reader_at_end(&reader) == (!test->remaining));
} else {
g_assert(!gbinder_reader_read_string16_utf16(&reader, NULL));
}
g_assert(gbinder_reader_at_end(&reader) == (!test->remaining));
g_assert(gbinder_reader_bytes_remaining(&reader) == test->remaining);
gbinder_reader_init(&reader, &data, 0, test->in_size);
g_assert(gbinder_reader_read_nullable_string16_utf16(&reader, NULL,
NULL) == valid);
@@ -939,6 +963,40 @@ test_hidl_string_err(
gbinder_ipc_unref(ipc);
}
static
void
test_hidl_string_err_skip(
gconstpointer test_data)
{
const TestHidlStringErr* test = test_data;
GBinderIpc* ipc = gbinder_ipc_new(GBINDER_DEFAULT_BINDER, NULL);
GBinderBuffer* buf = gbinder_buffer_new(ipc->driver,
g_memdup(test->in, test->in_size), test->in_size, NULL);
GBinderReaderData data;
GBinderReader reader;
g_assert(ipc);
memset(&data, 0, sizeof(data));
data.buffer = buf;
data.reg = gbinder_ipc_object_registry(ipc);
if (test->offset) {
guint i;
data.objects = g_new(void*, test->offset_count + 1);
for (i = 0; i < test->offset_count; i++) {
data.objects[i] = (guint8*)buf->data + test->offset[i];
}
data.objects[i] = NULL;
}
gbinder_reader_init(&reader, &data, 0, test->in_size);
g_assert(!gbinder_reader_skip_hidl_string(&reader));
g_free(data.objects);
gbinder_buffer_free(buf);
gbinder_ipc_unref(ipc);
}
/*==========================================================================*
* fd_ok
*==========================================================================*/
@@ -1189,6 +1247,324 @@ test_dupfd_badfd(
gbinder_ipc_unref(ipc);
}
/*==========================================================================*
* hidl_string
*==========================================================================*/
static
void
test_hidl_string(
const guint8* input,
gsize size,
const guint* offsets,
guint bufcount,
const char* result)
{
GBinderIpc* ipc = gbinder_ipc_new(GBINDER_DEFAULT_HWBINDER, NULL);
GBinderBuffer* buf = gbinder_buffer_new(ipc->driver, g_memdup(input, size),
size, NULL);
GBinderRemoteObject* obj = NULL;
GBinderReaderData data;
GBinderReader reader;
guint i;
g_assert(ipc);
memset(&data, 0, sizeof(data));
data.buffer = buf;
data.reg = gbinder_ipc_object_registry(ipc);
data.objects = g_new(void*, bufcount + 1);
for (i = 0; i < bufcount; i++) {
data.objects[i] = buf->data + offsets[i];
}
data.objects[i] = NULL;
gbinder_reader_init(&reader, &data, 0, buf->size);
g_assert(gbinder_reader_read_hidl_string_c(&reader) == result);
g_free(data.objects);
gbinder_remote_object_unref(obj);
gbinder_buffer_free(buf);
gbinder_ipc_unref(ipc);
}
static
void
test_hidl_string1(
void)
{
const char contents[] = "test";
const GBinderHidlString str = {
.data = { (uintptr_t)contents },
.len = sizeof(contents) - 1,
.owns_buffer = TRUE
};
/* Using 64-bit I/O */
const guint8 input[] = {
/* Buffer object #1 */
TEST_INT32_BYTES(BINDER_TYPE_PTR), TEST_INT32_BYTES(0),
TEST_INT64_BYTES((uintptr_t)&str), TEST_INT64_BYTES(sizeof(str)),
TEST_INT64_BYTES(0), TEST_INT64_BYTES(0),
/* Buffer object #2 */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(BINDER_BUFFER_FLAG_HAS_PARENT),
TEST_INT64_BYTES((uintptr_t)contents),
TEST_INT64_BYTES(sizeof(contents)),
TEST_INT64_BYTES(0),
TEST_INT64_BYTES(GBINDER_HIDL_STRING_BUFFER_OFFSET)
};
static const guint offsets[] = { 0, BUFFER_OBJECT_SIZE_64 };
test_hidl_string(TEST_ARRAY_AND_SIZE(input),
TEST_ARRAY_AND_COUNT(offsets), contents);
}
static
void
test_hidl_string2(
void)
{
const char contents[] = "test";
const GBinderHidlString str = {
.data = { (uintptr_t)contents },
.len = sizeof(contents) - 1,
.owns_buffer = TRUE
};
/* Using 64-bit I/O */
const guint8 input[] = {
/* Buffer object #1 */
TEST_INT32_BYTES(BINDER_TYPE_PTR), TEST_INT32_BYTES(0),
TEST_INT64_BYTES((uintptr_t)&str), TEST_INT64_BYTES(sizeof(str)),
TEST_INT64_BYTES(0), TEST_INT64_BYTES(0),
/* Invalid object type */
TEST_INT32_BYTES(BINDER_TYPE_HANDLE),
TEST_INT32_BYTES(BINDER_BUFFER_FLAG_HAS_PARENT),
TEST_INT64_BYTES((uintptr_t)contents),
TEST_INT64_BYTES(sizeof(contents)),
TEST_INT64_BYTES(0),
TEST_INT64_BYTES(GBINDER_HIDL_STRING_BUFFER_OFFSET)
};
static const guint offsets[] = { 0, BUFFER_OBJECT_SIZE_64 };
test_hidl_string(TEST_ARRAY_AND_SIZE(input),
TEST_ARRAY_AND_COUNT(offsets), NULL);
}
static
void
test_hidl_string3(
void)
{
const char contents[] = "test";
const GBinderHidlString str = {
.data = { (uintptr_t)contents },
.len = sizeof(contents) - 1,
.owns_buffer = TRUE
};
/* Using 64-bit I/O */
const guint8 input[] = {
/* Buffer object #1 */
TEST_INT32_BYTES(BINDER_TYPE_PTR), TEST_INT32_BYTES(0),
TEST_INT64_BYTES((uintptr_t)&str), TEST_INT64_BYTES(sizeof(str)),
TEST_INT64_BYTES(0), TEST_INT64_BYTES(0),
/* No parent */
TEST_INT32_BYTES(BINDER_TYPE_PTR), TEST_INT32_BYTES(0),
TEST_INT64_BYTES((uintptr_t)contents),
TEST_INT64_BYTES(sizeof(contents)),
TEST_INT64_BYTES(0),
TEST_INT64_BYTES(GBINDER_HIDL_STRING_BUFFER_OFFSET)
};
static const guint offsets[] = { 0, BUFFER_OBJECT_SIZE_64 };
test_hidl_string(TEST_ARRAY_AND_SIZE(input),
TEST_ARRAY_AND_COUNT(offsets), NULL);
}
static
void
test_hidl_string4(
void)
{
const char contents[] = "test";
const GBinderHidlString str = {
.data = { (uintptr_t)contents },
.len = sizeof(contents) - 1,
.owns_buffer = TRUE
};
/* Using 64-bit I/O */
const guint8 input[] = {
/* Buffer object #1 */
TEST_INT32_BYTES(BINDER_TYPE_PTR), TEST_INT32_BYTES(0),
TEST_INT64_BYTES((uintptr_t)&str), TEST_INT64_BYTES(sizeof(str)),
TEST_INT64_BYTES(0), TEST_INT64_BYTES(0),
/* Invalid length */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(BINDER_BUFFER_FLAG_HAS_PARENT),
TEST_INT64_BYTES((uintptr_t)contents),
TEST_INT64_BYTES(sizeof(contents) - 1),
TEST_INT64_BYTES(0),
TEST_INT64_BYTES(GBINDER_HIDL_STRING_BUFFER_OFFSET)
};
static const guint offsets[] = { 0, BUFFER_OBJECT_SIZE_64 };
test_hidl_string(TEST_ARRAY_AND_SIZE(input),
TEST_ARRAY_AND_COUNT(offsets), NULL);
}
static
void
test_hidl_string5(
void)
{
const char contents[] = "test";
const GBinderHidlString str = {
.data = { (uintptr_t)contents },
.len = sizeof(contents) - 1,
.owns_buffer = TRUE
};
/* Using 64-bit I/O */
const guint8 input[] = {
/* Buffer object #1 */
TEST_INT32_BYTES(BINDER_TYPE_PTR), TEST_INT32_BYTES(0),
TEST_INT64_BYTES((uintptr_t)&str), TEST_INT64_BYTES(sizeof(str)),
TEST_INT64_BYTES(0), TEST_INT64_BYTES(0),
/* Invalid pointer */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(BINDER_BUFFER_FLAG_HAS_PARENT),
TEST_INT64_BYTES((uintptr_t)contents + 1),
TEST_INT64_BYTES(sizeof(contents)),
TEST_INT64_BYTES(0),
TEST_INT64_BYTES(GBINDER_HIDL_STRING_BUFFER_OFFSET)
};
static const guint offsets[] = { 0, BUFFER_OBJECT_SIZE_64 };
test_hidl_string(TEST_ARRAY_AND_SIZE(input),
TEST_ARRAY_AND_COUNT(offsets), NULL);
}
static
void
test_hidl_string6(
void)
{
/* No NULL-terminated */
const char contents[] = "testx";
const GBinderHidlString str = {
.data = { (uintptr_t)contents },
.len = 4,
.owns_buffer = TRUE
};
/* Using 64-bit I/O */
const guint8 input[] = {
/* Buffer object #1 */
TEST_INT32_BYTES(BINDER_TYPE_PTR), TEST_INT32_BYTES(0),
TEST_INT64_BYTES((uintptr_t)&str), TEST_INT64_BYTES(sizeof(str)),
TEST_INT64_BYTES(0), TEST_INT64_BYTES(0),
/* Buffer object #2 */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(BINDER_BUFFER_FLAG_HAS_PARENT),
TEST_INT64_BYTES((uintptr_t)contents),
TEST_INT64_BYTES(5),
TEST_INT64_BYTES(0),
TEST_INT64_BYTES(GBINDER_HIDL_STRING_BUFFER_OFFSET)
};
static const guint offsets[] = { 0, BUFFER_OBJECT_SIZE_64 };
test_hidl_string(TEST_ARRAY_AND_SIZE(input),
TEST_ARRAY_AND_COUNT(offsets), NULL);
}
static
void
test_hidl_string7(
void)
{
const char contents[] = "test";
const GBinderHidlString str = {
.data = { (uintptr_t)contents },
.len = sizeof(contents) - 1,
.owns_buffer = TRUE
};
/* Using 64-bit I/O */
const guint8 input[] = {
/* Buffer object #1 */
TEST_INT32_BYTES(BINDER_TYPE_PTR), TEST_INT32_BYTES(0),
TEST_INT64_BYTES((uintptr_t)&str), TEST_INT64_BYTES(sizeof(str)),
TEST_INT64_BYTES(0), TEST_INT64_BYTES(0),
/* Invalid parent offset */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(BINDER_BUFFER_FLAG_HAS_PARENT),
TEST_INT64_BYTES((uintptr_t)contents),
TEST_INT64_BYTES(sizeof(contents)),
TEST_INT64_BYTES(0),
TEST_INT64_BYTES(GBINDER_HIDL_STRING_BUFFER_OFFSET + 1)
};
static const guint offsets[] = { 0, BUFFER_OBJECT_SIZE_64 };
test_hidl_string(TEST_ARRAY_AND_SIZE(input),
TEST_ARRAY_AND_COUNT(offsets), NULL);
}
/*==========================================================================*
* buffer
*==========================================================================*/
static
void
test_buffer(
void)
{
/* Using 64-bit I/O */
const int data1 = 0x1234;
const int data2 = 0x5678;
const guint8 input[] = {
/* Buffer object #1 */
TEST_INT32_BYTES(BINDER_TYPE_PTR), TEST_INT32_BYTES(0),
TEST_INT64_BYTES((uintptr_t)&data1), TEST_INT64_BYTES(sizeof(data1)),
TEST_INT64_BYTES(0), TEST_INT64_BYTES(0),
/* Buffer object #2 */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(BINDER_BUFFER_FLAG_HAS_PARENT),
TEST_INT64_BYTES((uintptr_t)&data2), TEST_INT64_BYTES(sizeof(data2)),
TEST_INT64_BYTES(0), TEST_INT64_BYTES(0),
/* Not a buffer object */
TEST_INT32_BYTES(BINDER_TYPE_HANDLE), TEST_INT32_BYTES(0),
TEST_INT64_BYTES(0), TEST_INT64_BYTES(0),
TEST_INT64_BYTES(0), TEST_INT64_BYTES(0)
};
GBinderIpc* ipc = gbinder_ipc_new(GBINDER_DEFAULT_HWBINDER, NULL);
GBinderBuffer* buf = gbinder_buffer_new(ipc->driver,
g_memdup(input, sizeof(input)), sizeof(input), NULL);
GBinderRemoteObject* obj = NULL;
GBinderReaderData data;
GBinderReader reader;
GBinderBuffer* res;
g_assert(ipc);
memset(&data, 0, sizeof(data));
data.buffer = buf;
data.reg = gbinder_ipc_object_registry(ipc);
data.objects = g_new(void*, 4);
data.objects[0] = buf->data;
data.objects[1] = buf->data + BUFFER_OBJECT_SIZE_64;
data.objects[2] = buf->data + 2 * BUFFER_OBJECT_SIZE_64;
data.objects[3] = NULL;
gbinder_reader_init(&reader, &data, 0, buf->size);
g_assert(gbinder_reader_skip_buffer(&reader));
res = gbinder_reader_read_buffer(&reader);
g_assert(res);
g_assert(res->data == &data2);
/* The next one is not a buffer object */
g_assert(!gbinder_reader_skip_buffer(&reader));
gbinder_buffer_free(res);
g_free(data.objects);
gbinder_remote_object_unref(obj);
gbinder_buffer_free(buf);
gbinder_ipc_unref(ipc);
}
/*==========================================================================*
* object
*==========================================================================*/
@@ -1337,6 +1713,290 @@ test_vec(
gbinder_ipc_unref(ipc);
}
/*==========================================================================*
* hidl_string_vec
*==========================================================================*/
static
void
test_hidl_string_vec(
const guint8* input,
gsize size,
const char* const* result)
{
GBinderIpc* ipc = gbinder_ipc_new(GBINDER_DEFAULT_HWBINDER, NULL);
GBinderBuffer* buf = gbinder_buffer_new(ipc->driver, g_memdup(input, size),
size, NULL);
GBinderRemoteObject* obj = NULL;
GBinderReaderData data;
GBinderReader reader;
char** out;
guint i;
g_assert(ipc);
memset(&data, 0, sizeof(data));
data.buffer = buf;
data.reg = gbinder_ipc_object_registry(ipc);
/* We assume that input consists only from buffer objects */
g_assert(!(size % BUFFER_OBJECT_SIZE_64));
data.objects = g_new(void*, size/BUFFER_OBJECT_SIZE_64 + 1);
for (i = 0; i < size/BUFFER_OBJECT_SIZE_64; i++) {
data.objects[i] = buf->data + i * BUFFER_OBJECT_SIZE_64;
}
data.objects[i] = NULL;
gbinder_reader_init(&reader, &data, 0, buf->size);
out = gbinder_reader_read_hidl_string_vec(&reader)
;
if (out) {
const guint n = g_strv_length(out);
g_assert(result);
g_assert(n == g_strv_length((char**)result));
for (i = 0; i < n; i++) {
g_assert(!g_strcmp0(out[i], result[i]));
}
} else {
g_assert(!result);
}
g_strfreev(out);
g_free(data.objects);
gbinder_remote_object_unref(obj);
gbinder_buffer_free(buf);
gbinder_ipc_unref(ipc);
}
static
void
test_hidl_string_vec1(
void)
{
static const char contents[] = "test";
const GBinderHidlString str = {
.data = { (uintptr_t)contents },
.len = sizeof(contents) - 1,
.owns_buffer = TRUE
};
const GBinderHidlVec vec = {
.data = { .ptr = &str },
.count = 1,
.owns_buffer = TRUE
};
/* Using 64-bit I/O */
const guint8 input[] = {
/* Buffer object #1 */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(0),
TEST_INT64_BYTES((uintptr_t)&vec),
TEST_INT64_BYTES(sizeof(vec)),
TEST_INT64_BYTES(0), TEST_INT64_BYTES(0),
/* Buffer object #2 */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(BINDER_BUFFER_FLAG_HAS_PARENT),
TEST_INT64_BYTES((uintptr_t)&str),
TEST_INT64_BYTES(sizeof(str)),
TEST_INT64_BYTES(1),
TEST_INT64_BYTES(GBINDER_HIDL_VEC_BUFFER_OFFSET),
/* Buffer object #3 */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(BINDER_BUFFER_FLAG_HAS_PARENT),
TEST_INT64_BYTES((uintptr_t)&contents),
TEST_INT64_BYTES(sizeof(contents)),
TEST_INT64_BYTES(2),
TEST_INT64_BYTES(GBINDER_HIDL_STRING_BUFFER_OFFSET)
};
static const char* const result[] = { contents, NULL };
test_hidl_string_vec(TEST_ARRAY_AND_SIZE(input), result);
}
static
void
test_hidl_string_vec2(
void)
{
static const char str1[] = "meh";
static const char str2[] = "foobar";
const GBinderHidlString str[] = {
{
.data = { (uintptr_t)str1 },
.len = sizeof(str1) - 1,
.owns_buffer = TRUE
},{
.data = { (uintptr_t)str2 },
.len = sizeof(str2) - 1,
.owns_buffer = TRUE
}
};
const GBinderHidlVec vec = {
.data = { .ptr = &str },
.count = 2,
.owns_buffer = TRUE
};
/* Using 64-bit I/O */
const guint8 input[] = {
/* Buffer object #1 */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(0),
TEST_INT64_BYTES((uintptr_t)&vec),
TEST_INT64_BYTES(sizeof(vec)),
TEST_INT64_BYTES(0), TEST_INT64_BYTES(0),
/* Buffer object #2 */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(BINDER_BUFFER_FLAG_HAS_PARENT),
TEST_INT64_BYTES((uintptr_t)str),
TEST_INT64_BYTES(sizeof(str)),
TEST_INT64_BYTES(1),
TEST_INT64_BYTES(GBINDER_HIDL_VEC_BUFFER_OFFSET),
/* Buffer object #3 */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(BINDER_BUFFER_FLAG_HAS_PARENT),
TEST_INT64_BYTES((uintptr_t)str1),
TEST_INT64_BYTES(sizeof(str1)),
TEST_INT64_BYTES(2),
TEST_INT64_BYTES(GBINDER_HIDL_STRING_BUFFER_OFFSET),
/* Buffer object #4 */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(BINDER_BUFFER_FLAG_HAS_PARENT),
TEST_INT64_BYTES((uintptr_t)str2),
TEST_INT64_BYTES(sizeof(str2)),
TEST_INT64_BYTES(2),
TEST_INT64_BYTES(sizeof(GBinderHidlString) +
GBINDER_HIDL_STRING_BUFFER_OFFSET)
};
static const char* const result[] = { str1, str2, NULL };
test_hidl_string_vec(TEST_ARRAY_AND_SIZE(input), result);
}
static
void
test_hidl_string_vec3(
void)
{
static const char contents[] = "test";
const GBinderHidlString str = {
.data = { (uintptr_t)contents },
.len = sizeof(contents) - 1,
.owns_buffer = TRUE
};
const GBinderHidlVec vec = {
.data = { .ptr = &str },
.count = 1,
.owns_buffer = TRUE
};
/* Using 64-bit I/O */
const guint8 input[] = {
/* Buffer object #1 */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(0),
TEST_INT64_BYTES((uintptr_t)&vec),
TEST_INT64_BYTES(sizeof(vec)),
TEST_INT64_BYTES(0), TEST_INT64_BYTES(0),
/* Buffer object #2 */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(BINDER_BUFFER_FLAG_HAS_PARENT),
TEST_INT64_BYTES((uintptr_t)&str),
TEST_INT64_BYTES(sizeof(str)),
TEST_INT64_BYTES(1),
TEST_INT64_BYTES(GBINDER_HIDL_VEC_BUFFER_OFFSET)
/* The next buffer is missing */
};
test_hidl_string_vec(TEST_ARRAY_AND_SIZE(input), NULL);
}
static
void
test_hidl_string_vec4(
void)
{
static const char contents[] = "test";
const GBinderHidlString str = {
.data = { (uintptr_t)contents },
.len = sizeof(contents) - 1,
.owns_buffer = TRUE
};
const GBinderHidlVec vec = {
.data = { .ptr = &str },
.count = 1,
.owns_buffer = TRUE
};
/* Using 64-bit I/O */
const guint8 input[] = {
/* Buffer object #1 */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(0),
TEST_INT64_BYTES((uintptr_t)&vec),
TEST_INT64_BYTES(sizeof(vec)),
TEST_INT64_BYTES(0), TEST_INT64_BYTES(0),
/* The next buffer is missing */
};
test_hidl_string_vec(TEST_ARRAY_AND_SIZE(input), NULL);
}
static
void
test_hidl_string_vec5(
void)
{
static const char str1[] = "meh";
static const char str2[] = "foobar";
const GBinderHidlString str[] = {
{
.data = { (uintptr_t)str1 },
.len = sizeof(str1) - 1,
.owns_buffer = TRUE
},{
.data = { (uintptr_t)str2 },
.len = sizeof(str2) - 1,
.owns_buffer = TRUE
}
};
const GBinderHidlVec vec = {
.data = { .ptr = &str },
.count = 2,
.owns_buffer = TRUE
};
/* Using 64-bit I/O */
const guint8 input[] = {
/* Buffer object #1 */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(0),
TEST_INT64_BYTES((uintptr_t)&vec),
TEST_INT64_BYTES(sizeof(vec)),
TEST_INT64_BYTES(0), TEST_INT64_BYTES(0),
/* Buffer object #2 */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(BINDER_BUFFER_FLAG_HAS_PARENT),
TEST_INT64_BYTES((uintptr_t)str),
TEST_INT64_BYTES(sizeof(str)),
TEST_INT64_BYTES(1),
TEST_INT64_BYTES(GBINDER_HIDL_VEC_BUFFER_OFFSET),
/* Buffer object #3 */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(BINDER_BUFFER_FLAG_HAS_PARENT),
TEST_INT64_BYTES((uintptr_t)str1),
TEST_INT64_BYTES(sizeof(str1)),
TEST_INT64_BYTES(2),
TEST_INT64_BYTES(GBINDER_HIDL_STRING_BUFFER_OFFSET),
/* Buffer object #4 (with invalid offset) */
TEST_INT32_BYTES(BINDER_TYPE_PTR),
TEST_INT32_BYTES(BINDER_BUFFER_FLAG_HAS_PARENT),
TEST_INT64_BYTES((uintptr_t)str2),
TEST_INT64_BYTES(sizeof(str2)),
TEST_INT64_BYTES(2),
TEST_INT64_BYTES(GBINDER_HIDL_STRING_BUFFER_OFFSET)
};
test_hidl_string_vec(TEST_ARRAY_AND_SIZE(input), NULL);
}
/*==========================================================================*
* byte_array
*==========================================================================*/
@@ -1548,6 +2208,11 @@ int main(int argc, char* argv[])
g_test_add_data_func(path, test, test_hidl_string_err);
g_free(path);
path = g_strconcat(TEST_("hidl_string/err-skip-"), test->name,
NULL);
g_test_add_data_func(path, test, test_hidl_string_err_skip);
g_free(path);
}
g_test_add_func(TEST_("fd/ok"), test_fd_ok);
@@ -1556,10 +2221,23 @@ int main(int argc, char* argv[])
g_test_add_func(TEST_("dupfd/ok"), test_dupfd_ok);
g_test_add_func(TEST_("dupfd/badtype"), test_dupfd_badtype);
g_test_add_func(TEST_("dupfd/badfd"), test_dupfd_badfd);
g_test_add_func(TEST_("hidl_string/1"), test_hidl_string1);
g_test_add_func(TEST_("hidl_string/2"), test_hidl_string2);
g_test_add_func(TEST_("hidl_string/3"), test_hidl_string3);
g_test_add_func(TEST_("hidl_string/4"), test_hidl_string4);
g_test_add_func(TEST_("hidl_string/5"), test_hidl_string5);
g_test_add_func(TEST_("hidl_string/6"), test_hidl_string6);
g_test_add_func(TEST_("hidl_string/7"), test_hidl_string7);
g_test_add_func(TEST_("buffer"), test_buffer);
g_test_add_func(TEST_("object/valid"), test_object);
g_test_add_func(TEST_("object/invalid"), test_object_invalid);
g_test_add_func(TEST_("object/no_reg"), test_object_no_reg);
g_test_add_func(TEST_("vec"), test_vec);
g_test_add_func(TEST_("hidl_string_vec/1"), test_hidl_string_vec1);
g_test_add_func(TEST_("hidl_string_vec/2"), test_hidl_string_vec2);
g_test_add_func(TEST_("hidl_string_vec/3"), test_hidl_string_vec3);
g_test_add_func(TEST_("hidl_string_vec/4"), test_hidl_string_vec4);
g_test_add_func(TEST_("hidl_string_vec/5"), test_hidl_string_vec5);
g_test_add_func(TEST_("byte_array"), test_byte_array);
g_test_add_func(TEST_("copy"), test_copy);
test_init(&test_opt, argc, argv);

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -50,7 +50,7 @@ void
test_null(
void)
{
g_assert(!gbinder_remote_object_new(NULL, 0));
g_assert(!gbinder_remote_object_new(NULL, 0, FALSE));
g_assert(!gbinder_remote_object_ref(NULL));
gbinder_remote_object_unref(NULL);
g_assert(gbinder_remote_object_is_dead(NULL));
@@ -80,7 +80,7 @@ test_basic(
g_assert(gbinder_remote_object_ref(obj1) == obj1);
gbinder_remote_object_unref(obj1); /* Compensate the above reference */
g_assert(!gbinder_remote_object_add_death_handler(obj1, NULL, NULL));
g_assert(gbinder_ipc_get_remote_object(ipc, 1) == obj1);
g_assert(gbinder_ipc_get_remote_object(ipc, 1, TRUE) == obj1);
gbinder_remote_object_unref(obj1); /* Compensate the above reference */
gbinder_remote_object_unref(obj1);
gbinder_remote_object_unref(obj2);
@@ -109,11 +109,14 @@ test_dead(
const guint handle = 1;
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
GBinderIpc* ipc = gbinder_ipc_new(GBINDER_DEFAULT_BINDER, NULL);
GBinderRemoteObject* obj = gbinder_ipc_get_remote_object(ipc, handle);
const int fd = gbinder_driver_fd(ipc->driver);
GBinderRemoteObject* obj = gbinder_ipc_get_remote_object
(ipc, handle, FALSE);
gulong id = gbinder_remote_object_add_death_handler
(obj, test_dead_done, loop);
test_binder_br_dead_binder(gbinder_driver_fd(ipc->driver), handle);
test_binder_br_dead_binder(fd, handle);
test_binder_set_looper_enabled(fd, TRUE);
test_run(&test_opt, loop);
g_assert(gbinder_remote_object_is_dead(obj));

View File

@@ -73,6 +73,8 @@ test_null(
gbinder_remote_request_unref(NULL);
gbinder_remote_request_set_data(NULL, 0, NULL);
gbinder_remote_request_init_reader(NULL, &reader);
gbinder_remote_request_block(NULL);
gbinder_remote_request_complete(NULL, NULL, 0);
g_assert(gbinder_reader_at_end(&reader));
g_assert(!gbinder_remote_request_interface(NULL));
g_assert(!gbinder_remote_request_copy_to_local(NULL));
@@ -100,6 +102,10 @@ test_basic(
GBinderRemoteRequest* req = gbinder_remote_request_new(NULL,
gbinder_rpc_protocol_for_device(NULL), 0, 0);
/* These two calls are wrong but won't cause problems: */
gbinder_remote_request_block(req);
gbinder_remote_request_complete(req, NULL, 0);
gbinder_remote_request_init_reader(req, &reader);
g_assert(gbinder_reader_at_end(&reader));
g_assert(!gbinder_remote_request_interface(req));

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -30,8 +30,9 @@
* THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "test_common.h"
#include "test_binder.h"
#include "gbinder_driver.h"
#include "gbinder_client_p.h"
#include "gbinder_remote_object_p.h"
#include "gbinder_ipc.h"
@@ -41,6 +42,7 @@
#include <gutil_strv.h>
#include <gutil_macros.h>
#include <gutil_log.h>
#include <errno.h>
@@ -92,6 +94,47 @@ test_transact_func(
return NULL;
}
static
void
test_inc(
GBinderServiceManager* sm,
void* user_data)
{
(*((int*)user_data))++;
}
static
void
test_reg_inc(
GBinderServiceManager* sm,
const char* name,
void* user_data)
{
GVERBOSE_("\"%s\"", name);
(*((int*)user_data))++;
}
static
void
test_quit(
GBinderServiceManager* sm,
void* user_data)
{
test_quit_later((GMainLoop*)user_data);
}
static
void
test_setup_ping(
GBinderIpc* ipc)
{
const int fd = gbinder_driver_fd(ipc->driver);
test_binder_br_noop(fd);
test_binder_br_transaction_complete(fd);
test_binder_br_reply(fd, 0, 0, NULL);
}
/*==========================================================================*
* TestServiceManager
*==========================================================================*/
@@ -131,7 +174,7 @@ test_servicemanager_get_service(
if (gutil_strv_contains(self->services, name)) {
if (!self->remote) {
self->remote = gbinder_ipc_get_remote_object
(gbinder_client_ipc(sm->client), 1);
(gbinder_client_ipc(sm->client), 1, TRUE);
}
*status = GBINDER_STATUS_OK;
return gbinder_remote_object_ref(self->remote);
@@ -355,6 +398,8 @@ test_null(
g_assert(!gbinder_servicemanager_new_with_type(0, NULL));
g_assert(!gbinder_servicemanager_new_local_object(NULL, NULL, NULL, NULL));
g_assert(!gbinder_servicemanager_ref(NULL));
g_assert(!gbinder_servicemanager_is_present(NULL));
g_assert(!gbinder_servicemanager_wait(NULL, 0));
g_assert(!gbinder_servicemanager_list(NULL, NULL, NULL));
g_assert(!gbinder_servicemanager_list_sync(NULL));
g_assert(!gbinder_servicemanager_get_service(NULL, NULL, NULL, NULL));
@@ -362,9 +407,11 @@ test_null(
g_assert(!gbinder_servicemanager_add_service(NULL, NULL, NULL, NULL, NULL));
g_assert(gbinder_servicemanager_add_service_sync(NULL, NULL, NULL) ==
(-EINVAL));
g_assert(!gbinder_servicemanager_add_presence_handler(NULL, NULL, NULL));
g_assert(!gbinder_servicemanager_add_registration_handler(NULL, NULL,
NULL, NULL));
gbinder_servicemanager_remove_handler(NULL, 0);
gbinder_servicemanager_remove_handlers(NULL, NULL, 0);
gbinder_servicemanager_cancel(NULL, 0);
gbinder_servicemanager_unref(NULL);
}
@@ -379,9 +426,13 @@ test_invalid(
void)
{
int status = 0;
GBinderServiceManager* sm =
gbinder_servicemanager_new(GBINDER_DEFAULT_HWBINDER);
const char* dev = GBINDER_DEFAULT_HWBINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
GBinderServiceManager* sm;
gulong id = 0;
test_setup_ping(ipc);
sm = gbinder_servicemanager_new(dev);
g_assert(!gbinder_servicemanager_new_with_type(GBINDER_TYPE_LOCAL_OBJECT,
NULL));
g_assert(TEST_IS_HWSERVICEMANAGER(sm));
@@ -399,12 +450,16 @@ test_invalid(
(-EINVAL));
g_assert(gbinder_servicemanager_add_service_sync(sm, "foo", NULL) ==
(-EINVAL));
g_assert(!gbinder_servicemanager_add_presence_handler(sm, NULL, NULL));
g_assert(!gbinder_servicemanager_add_registration_handler(sm, NULL, NULL,
NULL));
gbinder_servicemanager_cancel(sm, 0);
gbinder_servicemanager_remove_handler(sm, 0);
gbinder_servicemanager_remove_handlers(sm, NULL, 0);
gbinder_servicemanager_remove_handlers(sm, &id, 0);
gbinder_servicemanager_unref(sm);
gbinder_ipc_unref(ipc);
}
/*==========================================================================*
@@ -416,10 +471,13 @@ void
test_basic(
void)
{
GBinderServiceManager* sm =
gbinder_servicemanager_new(GBINDER_DEFAULT_HWBINDER);
const char* dev = GBINDER_DEFAULT_HWBINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
GBinderServiceManager* sm;
GBinderLocalObject* obj;
test_setup_ping(ipc);
sm = gbinder_servicemanager_new(dev);
g_assert(sm);
obj = gbinder_servicemanager_new_local_object(sm, "foo.bar",
test_transact_func, NULL);
@@ -429,6 +487,290 @@ test_basic(
g_assert(gbinder_servicemanager_ref(sm) == sm);
gbinder_servicemanager_unref(sm);
gbinder_servicemanager_unref(sm);
gbinder_ipc_unref(ipc);
}
/*==========================================================================*
* not_present
*==========================================================================*/
static
void
test_not_present(
void)
{
const char* dev = GBINDER_DEFAULT_HWBINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
const int fd = gbinder_driver_fd(ipc->driver);
GBinderServiceManager* sm;
/* This makes presence detection PING fail */
test_binder_br_reply_status(fd, -1);
sm = gbinder_servicemanager_new(dev);
g_assert(sm);
g_assert(!gbinder_servicemanager_is_present(sm));
gbinder_servicemanager_unref(sm);
gbinder_ipc_unref(ipc);
}
/*==========================================================================*
* wait
*==========================================================================*/
static
void
test_wait(
void)
{
const char* dev = GBINDER_DEFAULT_HWBINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
const int fd = gbinder_driver_fd(ipc->driver);
const glong forever = (test_opt.flags & TEST_FLAG_DEBUG) ?
(TEST_TIMEOUT_SEC * 1000) : -1;
GBinderServiceManager* sm;
gulong id;
int count = 0;
/* This makes presence detection PING fail */
test_binder_br_reply_status(fd, -1);
sm = gbinder_servicemanager_new(dev);
g_assert(sm);
g_assert(!gbinder_servicemanager_is_present(sm));
/* Register the listener */
id = gbinder_servicemanager_add_presence_handler(sm, test_inc, &count);
g_assert(id);
/* Make this wait fail */
test_binder_br_reply_status(fd, -1);
g_assert(!gbinder_servicemanager_wait(sm, 0));
/* This makes presence detection PING succeed */
test_binder_br_noop(fd);
test_binder_br_transaction_complete(fd);
test_binder_br_reply(fd, 0, 0, NULL);
g_assert(gbinder_servicemanager_wait(sm, forever));
/* The next check succeeds too (without any I/O ) */
g_assert(gbinder_servicemanager_is_present(sm));
g_assert(gbinder_servicemanager_wait(sm, 0));
/* The listener must have been invoked exactly once */
g_assert(count == 1);
gbinder_servicemanager_remove_handler(sm, id);
gbinder_servicemanager_unref(sm);
gbinder_ipc_unref(ipc);
}
/*==========================================================================*
* wait_long
*==========================================================================*/
static
void
test_wait_long(
void)
{
const char* dev = GBINDER_DEFAULT_HWBINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
const int fd = gbinder_driver_fd(ipc->driver);
GBinderServiceManager* sm;
gulong id;
int count = 0;
/* This makes presence detection PING fail */
test_binder_br_reply_status(fd, -1);
sm = gbinder_servicemanager_new(dev);
g_assert(sm);
g_assert(!gbinder_servicemanager_is_present(sm));
/* Register the listener */
id = gbinder_servicemanager_add_presence_handler(sm, test_inc, &count);
g_assert(id);
/* Make the first presence detection PING fail and second succeed */
test_binder_br_reply_status(fd, -1);
test_binder_br_reply_status_later(fd, -1);
test_binder_br_transaction_complete_later(fd);
test_binder_br_reply_later(fd, 0, 0, NULL);
g_assert(gbinder_servicemanager_wait(sm, TEST_TIMEOUT_SEC * 1000));
/* The next check succeeds too (without any I/O ) */
g_assert(gbinder_servicemanager_is_present(sm));
g_assert(gbinder_servicemanager_wait(sm, 0));
/* The listener must have been invoked exactly once */
g_assert(count == 1);
gbinder_servicemanager_remove_handler(sm, id);
gbinder_servicemanager_unref(sm);
gbinder_ipc_unref(ipc);
}
/*==========================================================================*
* wait_async
*==========================================================================*/
static
void
test_wait_async(
void)
{
const char* dev = GBINDER_DEFAULT_HWBINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
const int fd = gbinder_driver_fd(ipc->driver);
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
GBinderServiceManager* sm;
gulong id[2];
int count = 0;
/* This makes presence detection PING fail */
test_binder_br_reply_status(fd, -1);
sm = gbinder_servicemanager_new(dev);
g_assert(sm);
g_assert(!gbinder_servicemanager_is_present(sm));
/* Register the listeners */
id[0] = gbinder_servicemanager_add_presence_handler(sm, test_inc, &count);
id[1] = gbinder_servicemanager_add_presence_handler(sm, test_quit, loop);
g_assert(id[0]);
g_assert(id[1]);
/* Make the first presence detection PING fail and second succeed */
test_binder_br_reply_status(fd, -1);
test_binder_br_transaction_complete_later(fd);
test_binder_br_reply_later(fd, 0, 0, NULL);
test_run(&test_opt, loop);
/* The listener must have been invoked exactly once */
g_assert(count == 1);
gbinder_servicemanager_remove_all_handlers(sm, id);
gbinder_servicemanager_unref(sm);
gbinder_ipc_unref(ipc);
g_main_loop_unref(loop);
}
/*==========================================================================*
* death
*==========================================================================*/
static
void
test_death(
void)
{
const char* dev = GBINDER_DEFAULT_HWBINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
const int fd = gbinder_driver_fd(ipc->driver);
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
GBinderServiceManager* sm;
gulong id[3];
int count = 0, reg_count = 0;
test_setup_ping(ipc);
sm = gbinder_servicemanager_new(dev);
g_assert(sm);
g_assert(gbinder_servicemanager_is_present(sm));
/* Register the listeners */
id[0] = gbinder_servicemanager_add_presence_handler(sm, test_inc, &count);
id[1] = gbinder_servicemanager_add_presence_handler(sm, test_quit, loop);
id[2] = gbinder_servicemanager_add_registration_handler(sm, "foo",
test_reg_inc, &reg_count);
g_assert(id[0]);
g_assert(id[1]);
g_assert(id[2]);
/* Generate death notification (need looper for that) */
test_binder_br_dead_binder(fd, 0);
test_binder_set_looper_enabled(fd, TRUE);
test_run(&test_opt, loop);
/* No registrations must have occured */
g_assert(!reg_count);
/* The listener must have been invoked exactly once */
g_assert(count == 1);
g_assert(!gbinder_servicemanager_is_present(sm));
gbinder_servicemanager_remove_all_handlers(sm, id);
gbinder_servicemanager_unref(sm);
gbinder_ipc_unref(ipc);
g_main_loop_unref(loop);
}
/*==========================================================================*
* reanimate
*==========================================================================*/
static
void
test_reanimate_quit(
GBinderServiceManager* sm,
void* user_data)
{
if (gbinder_servicemanager_is_present(sm)) {
GDEBUG("Service manager is back");
test_quit_later((GMainLoop*)user_data);
} else {
const int fd = gbinder_driver_fd(sm->client->remote->ipc->driver);
/* Disable looper and reanimate the object */
GDEBUG("Reanimating...");
test_binder_set_looper_enabled(fd, FALSE);
test_binder_br_transaction_complete(fd);
test_binder_br_reply(fd, 0, 0, NULL);
}
}
static
void
test_reanimate(
void)
{
const char* dev = GBINDER_DEFAULT_HWBINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
const int fd = gbinder_driver_fd(ipc->driver);
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
GBinderServiceManager* sm;
gulong id[3];
int count = 0, reg_count = 0;
/* Create live service manager */
test_setup_ping(ipc);
sm = gbinder_servicemanager_new(dev);
g_assert(sm);
g_assert(gbinder_servicemanager_is_present(sm));
/* Register the listeners */
id[0] = gbinder_servicemanager_add_presence_handler(sm, test_inc, &count);
id[1] = gbinder_servicemanager_add_presence_handler(sm,
test_reanimate_quit, loop);
id[2] = gbinder_servicemanager_add_registration_handler(sm, "foo",
test_reg_inc, &reg_count);
g_assert(id[0]);
g_assert(id[1]);
g_assert(id[2]);
/* Generate death notification (need looper for that) */
test_binder_br_dead_binder(fd, 0);
test_binder_set_looper_enabled(fd, TRUE);
test_run(&test_opt, loop);
/* No registrations must have occured */
g_assert(!reg_count);
/* Presence must have changed twice */
g_assert(count == 2);
g_assert(gbinder_servicemanager_is_present(sm));
gbinder_servicemanager_remove_all_handlers(sm, id);
gbinder_servicemanager_unref(sm);
gbinder_ipc_unref(ipc);
g_main_loop_unref(loop);
}
/*==========================================================================*
@@ -440,18 +782,29 @@ void
test_reuse(
void)
{
GBinderServiceManager* m1 =
gbinder_servicemanager_new(GBINDER_DEFAULT_BINDER);
GBinderServiceManager* m2 =
gbinder_servicemanager_new(GBINDER_DEFAULT_BINDER);
GBinderServiceManager* vnd1 =
gbinder_servicemanager_new("/dev/vpnbinder");
GBinderServiceManager* vnd2 =
gbinder_servicemanager_new("/dev/vpnbinder");
GBinderServiceManager* hw1 =
gbinder_servicemanager_new(GBINDER_DEFAULT_HWBINDER);
GBinderServiceManager* hw2 =
gbinder_servicemanager_new(GBINDER_DEFAULT_HWBINDER);
const char* binder_dev = GBINDER_DEFAULT_BINDER;
const char* vndbinder_dev = "/dev/vpnbinder";
const char* hwbinder_dev = GBINDER_DEFAULT_HWBINDER;
GBinderIpc* binder_ipc = gbinder_ipc_new(binder_dev, NULL);
GBinderIpc* vndbinder_ipc = gbinder_ipc_new(vndbinder_dev, NULL);
GBinderIpc* hwbinder_ipc = gbinder_ipc_new(hwbinder_dev, NULL);
GBinderServiceManager* m1;
GBinderServiceManager* m2;
GBinderServiceManager* vnd1;
GBinderServiceManager* vnd2;
GBinderServiceManager* hw1;
GBinderServiceManager* hw2;
test_setup_ping(binder_ipc);
test_setup_ping(vndbinder_ipc);
test_setup_ping(hwbinder_ipc);
m1 = gbinder_servicemanager_new(binder_dev);
m2 = gbinder_servicemanager_new(binder_dev);
vnd1 = gbinder_servicemanager_new(vndbinder_dev);
vnd2 = gbinder_servicemanager_new(vndbinder_dev);
hw1 = gbinder_servicemanager_new(hwbinder_dev);
hw2 = gbinder_servicemanager_new(hwbinder_dev);
g_assert(m1);
g_assert(m1 == m2);
@@ -471,6 +824,9 @@ test_reuse(
gbinder_servicemanager_unref(vnd2);
gbinder_servicemanager_unref(hw1);
gbinder_servicemanager_unref(hw2);
gbinder_ipc_unref(binder_ipc);
gbinder_ipc_unref(vndbinder_ipc);
gbinder_ipc_unref(hwbinder_ipc);
}
/*==========================================================================*
@@ -480,15 +836,22 @@ test_reuse(
static
void
test_notify_type(
GType t)
GType t,
const char* dev)
{
GBinderServiceManager* sm = gbinder_servicemanager_new_with_type(t, NULL);
TestHwServiceManager* test = TEST_SERVICEMANAGER2(sm, t);
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
GBinderServiceManager* sm;
TestHwServiceManager* test;
const char* name = "foo";
int count = 0;
gulong id1 = gbinder_servicemanager_add_registration_handler(sm, name,
gulong id1, id2;
test_setup_ping(ipc);
sm = gbinder_servicemanager_new_with_type(t, NULL);
test = TEST_SERVICEMANAGER2(sm, t);
id1 = gbinder_servicemanager_add_registration_handler(sm, name,
test_registration_func_inc, &count);
gulong id2 = gbinder_servicemanager_add_registration_handler(sm, name,
id2 = gbinder_servicemanager_add_registration_handler(sm, name,
test_registration_func_inc, &count);
g_assert(id1 && id2);
@@ -507,6 +870,7 @@ test_notify_type(
gbinder_servicemanager_remove_handler(sm, id1);
gbinder_servicemanager_remove_handler(sm, id2);
gbinder_servicemanager_unref(sm);
gbinder_ipc_unref(ipc);
}
static
@@ -514,8 +878,8 @@ void
test_notify(
void)
{
test_notify_type(TEST_TYPE_HWSERVICEMANAGER);
test_notify_type(TEST_TYPE_DEFSERVICEMANAGER);
test_notify_type(TEST_TYPE_HWSERVICEMANAGER, GBINDER_DEFAULT_HWBINDER);
test_notify_type(TEST_TYPE_DEFSERVICEMANAGER, GBINDER_DEFAULT_BINDER);
}
/*==========================================================================*
@@ -541,12 +905,17 @@ void
test_list(
void)
{
GBinderServiceManager* sm = gbinder_servicemanager_new(NULL);
TestHwServiceManager* test = TEST_SERVICEMANAGER(sm);
const char* dev = GBINDER_DEFAULT_BINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
GBinderServiceManager* sm;
TestHwServiceManager* test;
char** list;
gulong id;
test_setup_ping(ipc);
sm = gbinder_servicemanager_new(dev);
test = TEST_SERVICEMANAGER(sm);
test->services = gutil_strv_add(test->services, "foo");
list = gbinder_servicemanager_list_sync(sm);
g_assert(gutil_strv_equal(test->services, list));
@@ -558,6 +927,7 @@ test_list(
test_run(&test_opt, loop);
gbinder_servicemanager_unref(sm);
gbinder_ipc_unref(ipc);
g_main_loop_unref(loop);
}
@@ -583,15 +953,21 @@ void
test_get(
void)
{
GBinderServiceManager* sm = gbinder_servicemanager_new(NULL);
TestHwServiceManager* test = TEST_SERVICEMANAGER(sm);
int status = -1;
GBinderLocalObject* obj =
gbinder_servicemanager_new_local_object(sm, "foo.bar",
test_transact_func, NULL);
const char* dev = GBINDER_DEFAULT_BINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
GBinderServiceManager* sm;
TestHwServiceManager* test;
int status = -1;
GBinderLocalObject* obj;
gulong id;
test_setup_ping(ipc);
sm = gbinder_servicemanager_new(dev);
test = TEST_SERVICEMANAGER(sm);
obj = gbinder_servicemanager_new_local_object(sm, "foo.bar",
test_transact_func, NULL);
/* Add a service */
g_assert(obj);
g_assert(gbinder_servicemanager_add_service_sync(sm, "foo", obj) ==
@@ -614,6 +990,7 @@ test_get(
test_run(&test_opt, loop);
gbinder_servicemanager_unref(sm);
gbinder_ipc_unref(ipc);
g_main_loop_unref(loop);
}
@@ -637,13 +1014,21 @@ void
test_add(
void)
{
GBinderServiceManager* sm = gbinder_servicemanager_new(NULL);
TestHwServiceManager* test = TEST_SERVICEMANAGER(sm);
GBinderLocalObject* obj =
gbinder_servicemanager_new_local_object(sm, "foo.bar",
test_transact_func, NULL);
const char* dev = GBINDER_DEFAULT_BINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
gulong id = gbinder_servicemanager_add_service(sm, "foo", obj,
GBinderServiceManager* sm;
TestHwServiceManager* test;
GBinderLocalObject* obj;
gulong id;
test_setup_ping(ipc);
sm = gbinder_servicemanager_new(dev);
test = TEST_SERVICEMANAGER(sm);
obj = gbinder_servicemanager_new_local_object(sm, "foo.bar",
test_transact_func, NULL);
id = gbinder_servicemanager_add_service(sm, "foo", obj,
test_add_func, loop);
g_assert(id);
@@ -651,6 +1036,7 @@ test_add(
g_assert(gutil_strv_contains(test->services, "foo"));
gbinder_servicemanager_unref(sm);
gbinder_ipc_unref(ipc);
g_main_loop_unref(loop);
}
@@ -666,6 +1052,12 @@ int main(int argc, char* argv[])
g_test_add_func(TEST_("null"), test_null);
g_test_add_func(TEST_("invalid"), test_invalid);
g_test_add_func(TEST_("basic"), test_basic);
g_test_add_func(TEST_("not_present"), test_not_present);
g_test_add_func(TEST_("wait"), test_wait);
g_test_add_func(TEST_("wait_long"), test_wait_long);
g_test_add_func(TEST_("wait_async"), test_wait_async);
g_test_add_func(TEST_("death"), test_death);
g_test_add_func(TEST_("reanimate"), test_reanimate);
g_test_add_func(TEST_("reuse"), test_reuse);
g_test_add_func(TEST_("notify"), test_notify);
g_test_add_func(TEST_("list"), test_list);

View File

@@ -0,0 +1,5 @@
# -*- Mode: makefile-gmake -*-
EXE = unit_servicename
include ../common/Makefile

View File

@@ -0,0 +1,499 @@
/*
* Copyright (C) 2019 Jolla Ltd.
* Copyright (C) 2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the names of the copyright holders nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "test_binder.h"
#include "gbinder_servicename.h"
#include "gbinder_servicemanager_p.h"
#include "gbinder_local_object.h"
#include "gbinder_rpc_protocol.h"
#include "gbinder_driver.h"
#include "gbinder_ipc.h"
#include <gutil_strv.h>
#include <gutil_log.h>
#include <errno.h>
static TestOpt test_opt;
static
void
test_quit(
GBinderServiceManager* sm,
void* user_data)
{
test_quit_later((GMainLoop*)user_data);
}
static
void
test_quit_when_destroyed(
gpointer loop,
GObject* obj)
{
test_quit_later((GMainLoop*)loop);
}
static
void
test_setup_ping(
GBinderIpc* ipc)
{
const int fd = gbinder_driver_fd(ipc->driver);
test_binder_br_noop(fd);
test_binder_br_transaction_complete(fd);
test_binder_br_reply(fd, 0, 0, NULL);
}
/*==========================================================================*
* TestServiceManager
*==========================================================================*/
typedef GBinderServiceManagerClass TestServiceManagerClass;
typedef struct test_servicemanager {
GBinderServiceManager manager;
GCond cond;
GMutex mutex;
char** services;
gboolean block_add;
int add_result;
} TestServiceManager;
G_DEFINE_TYPE(TestServiceManager, test_servicemanager,
GBINDER_TYPE_SERVICEMANAGER)
#define TEST_SERVICEMANAGER_HANDLE (0)
#define TEST_SERVICEMANAGER_IFACE "android.os.IServiceManager"
#define TEST_TYPE_SERVICEMANAGER (test_servicemanager_get_type())
#define TEST_SERVICEMANAGER(obj) G_TYPE_CHECK_INSTANCE_CAST((obj), \
TEST_TYPE_SERVICEMANAGER, TestServiceManager)
static
char**
test_servicemanager_list(
GBinderServiceManager* manager)
{
char** ret;
TestServiceManager* self = TEST_SERVICEMANAGER(manager);
g_mutex_lock(&self->mutex);
ret = g_strdupv(self->services);
GDEBUG("%u", gutil_strv_length(ret));
g_mutex_unlock(&self->mutex);
return ret;
}
static
GBinderRemoteObject*
test_servicemanager_get_service(
GBinderServiceManager* manager,
const char* name,
int* status)
{
*status = (-ENOENT);
return NULL;
}
static
int
test_servicemanager_add_service(
GBinderServiceManager* manager,
const char* name,
GBinderLocalObject* obj)
{
TestServiceManager* self = TEST_SERVICEMANAGER(manager);
g_mutex_lock(&self->mutex);
if (!gutil_strv_contains(self->services, name)) {
self->services = gutil_strv_add(self->services, name);
}
while (self->block_add) {
g_cond_wait(&self->cond, &self->mutex);
}
g_mutex_unlock(&self->mutex);
return self->add_result;
}
static
GBINDER_SERVICEMANAGER_NAME_CHECK
test_servicemanager_check_name(
GBinderServiceManager* manager,
const char* name)
{
return name ?
GBINDER_SERVICEMANAGER_NAME_INVALID :
GBINDER_SERVICEMANAGER_NAME_OK;
}
static
gboolean
test_servicemanager_watch(
GBinderServiceManager* manager,
const char* name)
{
return TRUE;
}
static
void
test_servicemanager_unwatch(
GBinderServiceManager* manager,
const char* name)
{
}
static
void
test_servicemanager_init(
TestServiceManager* self)
{
g_cond_init(&self->cond);
g_mutex_init(&self->mutex);
self->add_result = GBINDER_STATUS_OK;
}
static
void
test_servicemanager_finalize(
GObject* object)
{
TestServiceManager* self = TEST_SERVICEMANAGER(object);
g_cond_clear(&self->cond);
g_mutex_clear(&self->mutex);
g_strfreev(self->services);
G_OBJECT_CLASS(test_servicemanager_parent_class)->finalize(object);
}
static
void
test_servicemanager_class_init(
TestServiceManagerClass* klass)
{
klass->handle = TEST_SERVICEMANAGER_HANDLE;
klass->iface = TEST_SERVICEMANAGER_IFACE;
klass->default_device = GBINDER_DEFAULT_HWBINDER;
klass->rpc_protocol = &gbinder_rpc_protocol_binder;
klass->list = test_servicemanager_list;
klass->get_service = test_servicemanager_get_service;
klass->add_service = test_servicemanager_add_service;
klass->check_name = test_servicemanager_check_name;
klass->watch = test_servicemanager_watch;
klass->unwatch = test_servicemanager_unwatch;
G_OBJECT_CLASS(klass)->finalize = test_servicemanager_finalize;
}
GBinderServiceManager*
gbinder_defaultservicemanager_new(
const char* dev)
{
return gbinder_servicemanager_new_with_type(TEST_TYPE_SERVICEMANAGER, dev);
}
GBinderServiceManager*
gbinder_hwservicemanager_new(
const char* dev)
{
return gbinder_servicemanager_new(dev);
}
/*==========================================================================*
* null
*==========================================================================*/
static
void
test_null(
void)
{
const char* dev = GBINDER_DEFAULT_BINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
GBinderServiceManager* sm;
test_setup_ping(ipc);
sm = gbinder_servicemanager_new(dev);
g_assert(!gbinder_servicename_new(NULL, NULL, NULL));
g_assert(!gbinder_servicename_new(sm, NULL, NULL));
g_assert(!gbinder_servicename_ref(NULL));
gbinder_servicename_unref(NULL);
gbinder_servicemanager_unref(sm);
gbinder_ipc_unref(ipc);
}
/*==========================================================================*
* basic
*==========================================================================*/
static
void
test_basic(
void)
{
const char* obj_name = "test";
const char* dev = GBINDER_DEFAULT_BINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
GBinderLocalObject* obj;
GBinderServiceManager* sm;
GBinderServiceName* sn;
test_setup_ping(ipc);
sm = gbinder_servicemanager_new(dev);
obj = gbinder_ipc_new_local_object(ipc, "interface", NULL, NULL);
g_assert(!gbinder_servicename_new(sm, obj, NULL));
sn = gbinder_servicename_new(sm, obj, obj_name);
g_assert(sn);
g_assert(!g_strcmp0(sn->name, obj_name));
g_assert(gbinder_servicename_ref(sn) == sn);
gbinder_servicename_unref(sn);
gbinder_servicename_unref(sn);
gbinder_local_object_unref(obj);
gbinder_servicemanager_unref(sm);
/* We need to wait until GBinderIpc is destroyed */
GDEBUG("waiting for GBinderIpc to get destroyed");
g_object_weak_ref(G_OBJECT(ipc), test_quit_when_destroyed, loop);
gbinder_ipc_unref(ipc);
test_run(&test_opt, loop);
g_main_loop_unref(loop);
}
/*==========================================================================*
* present
*==========================================================================*/
static
void
test_present(
int add_result)
{
const char* obj_name = "test";
const char* dev = GBINDER_DEFAULT_BINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
const int fd = gbinder_driver_fd(ipc->driver);
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
GBinderLocalObject* obj;
GBinderServiceManager* sm;
GBinderServiceName* sn;
gulong id;
test_setup_ping(ipc);
sm = gbinder_servicemanager_new(dev);
TEST_SERVICEMANAGER(sm)->add_result = add_result;
obj = gbinder_ipc_new_local_object(ipc, "interface", NULL, NULL);
sn = gbinder_servicename_new(sm, obj, obj_name);
g_assert(sn);
g_assert(!g_strcmp0(sn->name, obj_name));
/* Immediately generate death notification (need looper for that) */
test_binder_br_dead_binder(fd, 0);
test_binder_set_looper_enabled(fd, TRUE);
id = gbinder_servicemanager_add_presence_handler(sm, test_quit, loop);
test_run(&test_opt, loop);
gbinder_servicename_unref(sn);
gbinder_local_object_unref(obj);
gbinder_servicemanager_remove_handler(sm, id);
gbinder_servicemanager_unref(sm);
/* We need to wait until GBinderIpc is destroyed */
GDEBUG("waiting for GBinderIpc to get destroyed");
g_object_weak_ref(G_OBJECT(ipc), test_quit_when_destroyed, loop);
gbinder_ipc_unref(ipc);
test_run(&test_opt, loop);
g_main_loop_unref(loop);
}
static
void
test_present_ok(
void)
{
test_present(GBINDER_STATUS_OK);
}
static
void
test_present_err(
void)
{
test_present(-1);
}
/*==========================================================================*
* not_present
*==========================================================================*/
static
void
test_not_present(
void)
{
const char* obj_name = "test";
const char* dev = GBINDER_DEFAULT_BINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
const int fd = gbinder_driver_fd(ipc->driver);
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
GBinderLocalObject* obj;
GBinderServiceManager* sm;
GBinderServiceName* sn;
gulong id;
/* This makes presence detection PING fail */
test_binder_br_reply_status(fd, -1);
sm = gbinder_servicemanager_new(dev);
g_assert(!gbinder_servicemanager_is_present(sm));
id = gbinder_servicemanager_add_presence_handler(sm, test_quit, loop);
obj = gbinder_ipc_new_local_object(ipc, "interface", NULL, NULL);
sn = gbinder_servicename_new(sm, obj, obj_name);
g_assert(sn);
g_assert(!g_strcmp0(sn->name, obj_name));
/* Make the next presence detection PING succeed */
test_binder_br_transaction_complete_later(fd);
test_binder_br_reply_later(fd, 0, 0, NULL);
test_run(&test_opt, loop);
gbinder_servicename_unref(sn);
gbinder_local_object_unref(obj);
gbinder_servicemanager_remove_handler(sm, id);
gbinder_servicemanager_unref(sm);
/* We need to wait until GBinderIpc is destroyed */
GDEBUG("waiting for GBinderIpc to get destroyed");
g_object_weak_ref(G_OBJECT(ipc), test_quit_when_destroyed, loop);
gbinder_ipc_unref(ipc);
test_run(&test_opt, loop);
g_main_loop_unref(loop);
}
/*==========================================================================*
* cancel
*==========================================================================*/
static
void
test_cancel(
void)
{
const char* obj_name = "test";
const char* dev = GBINDER_DEFAULT_BINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
const int fd = gbinder_driver_fd(ipc->driver);
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
GBinderLocalObject* obj;
GBinderServiceManager* sm;
TestServiceManager* test;
GBinderServiceName* sn;
gulong id;
test_setup_ping(ipc);
sm = gbinder_servicemanager_new(dev);
obj = gbinder_ipc_new_local_object(ipc, "interface", NULL, NULL);
/* Block name add calls */
test = TEST_SERVICEMANAGER(sm);
g_mutex_lock(&test->mutex);
test->block_add = TRUE;
g_mutex_unlock(&test->mutex);
/* This adds the name but the call blocks */
sn = gbinder_servicename_new(sm, obj, obj_name);
g_assert(sn);
g_assert(!g_strcmp0(sn->name, obj_name));
/* Immediately generate death notification (need looper for that) */
test_binder_br_dead_binder(fd, 0);
test_binder_set_looper_enabled(fd, TRUE);
id = gbinder_servicemanager_add_presence_handler(sm, test_quit, loop);
test_run(&test_opt, loop);
/* Add call is supposed to be cancelled */
gbinder_servicename_unref(sn);
gbinder_local_object_unref(obj);
gbinder_servicemanager_remove_handler(sm, id);
gbinder_servicemanager_unref(sm);
/* Unblock pending add */
g_mutex_lock(&test->mutex);
test->block_add = FALSE;
g_cond_signal(&test->cond);
g_mutex_unlock(&test->mutex);
/* We need to wait until GBinderIpc is destroyed */
GDEBUG("waiting for GBinderIpc to get destroyed");
g_object_weak_ref(G_OBJECT(ipc), test_quit_when_destroyed, loop);
gbinder_ipc_unref(ipc);
test_run(&test_opt, loop);
g_main_loop_unref(loop);
}
/*==========================================================================*
* Common
*==========================================================================*/
#define TEST_(test) "/servicename/" test
int main(int argc, char* argv[])
{
g_test_init(&argc, &argv, NULL);
g_test_add_func(TEST_("null"), test_null);
g_test_add_func(TEST_("basic"), test_basic);
g_test_add_func(TEST_("present_ok"), test_present_ok);
g_test_add_func(TEST_("present_err"), test_present_err);
g_test_add_func(TEST_("not_present"), test_not_present);
g_test_add_func(TEST_("cancel"), test_cancel);
test_init(&test_opt, argc, argv);
return g_test_run();
}
/*
* Local Variables:
* mode: C
* c-basic-offset: 4
* indent-tabs-mode: nil
* End:
*/

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -30,8 +30,10 @@
* THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "test_common.h"
#include "test_binder.h"
#include "gbinder_driver.h"
#include "gbinder_ipc.h"
#include "gbinder_servicemanager_p.h"
#include "gbinder_servicepoll.h"
#include "gbinder_rpc_protocol.h"
@@ -43,6 +45,18 @@
static TestOpt test_opt;
static
void
test_setup_ping(
GBinderIpc* ipc)
{
const int fd = gbinder_driver_fd(ipc->driver);
test_binder_br_noop(fd);
test_binder_br_transaction_complete(fd);
test_binder_br_reply(fd, 0, 0, NULL);
}
/*==========================================================================*
* TestServiceManager
*==========================================================================*/
@@ -212,10 +226,15 @@ void
test_basic(
void)
{
const char* dev = GBINDER_DEFAULT_BINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
GBinderServicePoll* weakptr = NULL;
GBinderServiceManager* manager = gbinder_servicemanager_new(NULL);
GBinderServicePoll* poll = gbinder_servicepoll_new(manager, NULL);
GBinderServiceManager* manager;
GBinderServicePoll* poll;
test_setup_ping(ipc);
manager = gbinder_servicemanager_new(dev);
poll = gbinder_servicepoll_new(manager, NULL);
g_assert(poll);
g_assert(gbinder_servicepoll_manager(poll) == manager);
g_assert(!gbinder_servicepoll_is_known_name(poll, "foo"));
@@ -230,6 +249,7 @@ test_basic(
gbinder_servicepoll_unref(poll);
gbinder_servicemanager_unref(manager);
gbinder_ipc_unref(ipc);
}
/*==========================================================================*
@@ -282,13 +302,19 @@ void
test_notify1(
void)
{
GBinderServicePoll* weakptr = NULL;
GBinderServiceManager* manager = gbinder_servicemanager_new(NULL);
TestServiceManager* test = TEST_SERVICEMANAGER(manager);
const char* dev = GBINDER_DEFAULT_BINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
GBinderServicePoll* weakptr = NULL;
GBinderServiceManager* manager;
TestServiceManager* test;
GBinderServicePoll* poll;
gulong id;
test_setup_ping(ipc);
manager = gbinder_servicemanager_new(dev);
test = TEST_SERVICEMANAGER(manager);
gbinder_servicepoll_interval_ms = 100;
poll = gbinder_servicepoll_new(manager, &weakptr);
g_timeout_add(2 * gbinder_servicepoll_interval_ms,
@@ -307,6 +333,7 @@ test_notify1(
gbinder_servicepoll_unref(poll);
g_assert(!weakptr);
gbinder_servicemanager_unref(manager);
gbinder_ipc_unref(ipc);
g_main_loop_unref(loop);
}
@@ -349,13 +376,19 @@ void
test_notify2(
void)
{
GBinderServicePoll* weakptr = NULL;
GBinderServiceManager* manager = gbinder_servicemanager_new(NULL);
TestServiceManager* test = TEST_SERVICEMANAGER(manager);
const char* dev = GBINDER_DEFAULT_BINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
GBinderServicePoll* weakptr = NULL;
GBinderServiceManager* manager;
TestServiceManager* test;
GBinderServicePoll* poll;
gulong id;
test_setup_ping(ipc);
manager = gbinder_servicemanager_new(dev);
test = TEST_SERVICEMANAGER(manager);
gbinder_servicepoll_interval_ms = 100;
poll = gbinder_servicepoll_new(manager, &weakptr);
g_timeout_add(2 * gbinder_servicepoll_interval_ms,
@@ -378,6 +411,7 @@ test_notify2(
gbinder_servicepoll_unref(poll);
g_assert(!weakptr);
gbinder_servicemanager_unref(manager);
gbinder_ipc_unref(ipc);
g_main_loop_unref(loop);
}
@@ -401,13 +435,20 @@ void
test_already_there(
void)
{
GBinderServicePoll* weakptr = NULL;
GBinderServiceManager* manager = gbinder_servicemanager_new(NULL);
GBinderServicePoll* poll = gbinder_servicepoll_new(manager, &weakptr);
TestServiceManager* test = TEST_SERVICEMANAGER(manager);
const char* dev = GBINDER_DEFAULT_BINDER;
GBinderIpc* ipc = gbinder_ipc_new(dev, NULL);
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
GBinderServicePoll* weakptr = NULL;
GBinderServiceManager* manager;
TestServiceManager* test;
GBinderServicePoll* poll;
gulong id;
test_setup_ping(ipc);
manager = gbinder_servicemanager_new(dev);
poll = gbinder_servicepoll_new(manager, &weakptr);
test = TEST_SERVICEMANAGER(manager);
test->services = gutil_strv_add(test->services, "foo");
id = gbinder_servicepoll_add_handler(poll, test_already_there_proc, loop);
@@ -418,6 +459,7 @@ test_already_there(
gbinder_servicepoll_unref(poll);
g_assert(!weakptr);
gbinder_servicemanager_unref(manager);
gbinder_ipc_unref(ipc);
g_main_loop_unref(loop);
}

View File

@@ -1,6 +1,6 @@
/*
* Copyright (C) 2018 Jolla Ltd.
* Copyright (C) 2018 Slava Monich <slava.monich@jolla.com>
* Copyright (C) 2018-2019 Jolla Ltd.
* Copyright (C) 2018-2019 Slava Monich <slava.monich@jolla.com>
*
* You may use this file under the terms of BSD license as follows:
*
@@ -97,9 +97,51 @@ test_null(
gbinder_writer_append_remote_object(&writer, NULL);
gbinder_writer_append_byte_array(NULL, NULL, 0);
gbinder_writer_append_byte_array(&writer, NULL, 0);
gbinder_writer_add_cleanup(NULL, NULL, 0);
gbinder_writer_add_cleanup(NULL, g_free, 0);
gbinder_writer_overwrite_int32(NULL, 0, 0);
g_assert(!gbinder_output_data_offsets(NULL));
g_assert(!gbinder_output_data_buffers_size(NULL));
g_assert(!gbinder_writer_malloc(NULL, 0));
g_assert(!gbinder_writer_malloc0(NULL, 0));
g_assert(!gbinder_writer_memdup(&writer, NULL, 0));
g_assert(!gbinder_writer_memdup(NULL, &writer, 0));
}
/*==========================================================================*
* cleanup
*==========================================================================*/
static
void
test_cleanup_fn(
gpointer ptr)
{
(*((int*)ptr))++;
}
static
void
test_cleanup(
void)
{
GBinderLocalRequest* req = gbinder_local_request_new(&gbinder_io_32, NULL);
GBinderWriter writer;
int cleanup_count = 0;
int value = 42;
int* zero;
int* copy;
gbinder_local_request_init_writer(req, &writer);
zero = gbinder_writer_new0(&writer, int);
copy = gbinder_writer_memdup(&writer, &value, sizeof(value));
g_assert(*zero == 0);
g_assert(*copy == value);
gbinder_writer_add_cleanup(&writer, test_cleanup_fn, &cleanup_count);
gbinder_writer_add_cleanup(&writer, test_cleanup_fn, &cleanup_count);
gbinder_local_request_unref(req);
g_assert(cleanup_count == 2);
}
/*==========================================================================*
@@ -123,6 +165,19 @@ test_int32(
g_assert(!gbinder_output_data_buffers_size(data));
g_assert(data->bytes->len == sizeof(value));
g_assert(!memcmp(data->bytes->data, &value, data->bytes->len));
const guint32 value2 = 2345678;
gbinder_writer_overwrite_int32(&writer, 0, value2);
data = gbinder_local_request_data(req);
g_assert(!gbinder_output_data_offsets(data));
g_assert(!gbinder_output_data_buffers_size(data));
g_assert(data->bytes->len == sizeof(value2));
g_assert(!memcmp(data->bytes->data, &value2, data->bytes->len));
// test overlap over the end of the buffer
gbinder_writer_overwrite_int32(&writer, 2, value2);
g_assert(data->bytes->len == sizeof(value2));
gbinder_local_request_unref(req);
}
@@ -883,6 +938,26 @@ test_byte_array(
gbinder_local_request_unref(req);
}
/*==========================================================================*
* bytes_written
*==========================================================================*/
static
void
test_bytes_written(
void)
{
const guint32 value = 1234567;
GBinderLocalRequest* req = gbinder_local_request_new(&gbinder_io_32, NULL);
GBinderWriter writer;
gbinder_local_request_init_writer(req, &writer);
g_assert(gbinder_writer_bytes_written(&writer) == 0);
gbinder_writer_append_int32(&writer, value);
g_assert(gbinder_writer_bytes_written(&writer) == sizeof(value));
gbinder_local_request_unref(req);
}
/*==========================================================================*
* Common
*==========================================================================*/
@@ -896,6 +971,7 @@ int main(int argc, char* argv[])
g_test_init(&argc, &argv, NULL);
g_test_add_func(TEST_("null"), test_null);
g_test_add_func(TEST_("cleanup"), test_cleanup);
g_test_add_func(TEST_("int32"), test_int32);
g_test_add_func(TEST_("int64"), test_int64);
g_test_add_func(TEST_("float"), test_float);
@@ -953,6 +1029,7 @@ int main(int argc, char* argv[])
g_test_add_func(TEST_("local_object"), test_local_object);
g_test_add_func(TEST_("remote_object"), test_remote_object);
g_test_add_func(TEST_("byte_array"), test_byte_array);
g_test_add_func(TEST_("bytes_written"), test_bytes_written);
test_init(&test_opt, argc, argv);
return g_test_run();
}