libgbinder/unit/unit_client/unit_client.c

457 lines
13 KiB
C

/*
* Copyright (C) 2018-2021 Jolla Ltd.
* Copyright (C) 2018-2021 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_client_p.h"
#include "gbinder_driver.h"
#include "gbinder_ipc.h"
#include "gbinder_local_reply_p.h"
#include "gbinder_local_request.h"
#include "gbinder_object_registry.h"
#include "gbinder_output_data.h"
#include "gbinder_remote_object_p.h"
#include "gbinder_remote_reply.h"
#include <gutil_log.h>
#include <errno.h>
static TestOpt test_opt;
static
GBinderClient*
test_client_new(
guint h,
const char* iface)
{
GBinderIpc* ipc = gbinder_ipc_new(GBINDER_DEFAULT_BINDER);
GBinderObjectRegistry* reg = gbinder_ipc_object_registry(ipc);
GBinderRemoteObject* obj = gbinder_object_registry_get_remote(reg, h, TRUE);
GBinderClient* client = gbinder_client_new(obj, iface);
g_assert(client);
gbinder_remote_object_unref(obj);
gbinder_ipc_unref(ipc);
return client;
}
/*==========================================================================*
* null
*==========================================================================*/
static
void
test_null(
void)
{
g_assert(!gbinder_client_new(NULL, NULL));
g_assert(!gbinder_client_new2(NULL, NULL, 0));
g_assert(!gbinder_client_ref(NULL));
g_assert(!gbinder_client_interface(NULL));
g_assert(!gbinder_client_interface2(NULL, 0));
gbinder_client_unref(NULL);
g_assert(!gbinder_client_new_request(NULL));
g_assert(!gbinder_client_new_request2(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);
}
/*==========================================================================*
* basic
*==========================================================================*/
static
void
test_basic(
void)
{
GBinderIpc* ipc = gbinder_ipc_new(GBINDER_DEFAULT_BINDER);
GBinderObjectRegistry* reg = gbinder_ipc_object_registry(ipc);
GBinderRemoteObject* obj = gbinder_object_registry_get_remote(reg, 0, TRUE);
const char* iface = "foo";
GBinderClient* client = gbinder_client_new(obj, iface);
g_assert(client);
g_assert(gbinder_client_ref(client) == client);
g_assert_cmpstr(gbinder_client_interface(client), == ,iface);
gbinder_client_unref(client);
gbinder_client_cancel(client, 0); /* does nothing */
gbinder_client_unref(client);
gbinder_remote_object_unref(obj);
gbinder_ipc_unref(ipc);
}
/*==========================================================================*
* interfaces
*==========================================================================*/
static
void
test_interfaces(
void)
{
GBinderIpc* ipc = gbinder_ipc_new(GBINDER_DEFAULT_BINDER);
GBinderObjectRegistry* reg = gbinder_ipc_object_registry(ipc);
GBinderRemoteObject* obj = gbinder_object_registry_get_remote(reg, 0, TRUE);
static const GBinderClientIfaceInfo ifaces[] = {
{"33", 33 }, { "11", 11 }, { "22", 22 }
};
GBinderClient* client = gbinder_client_new2(obj, ifaces,
G_N_ELEMENTS(ifaces));
g_assert(client);
g_assert_cmpstr(gbinder_client_interface(client), == ,"11");
g_assert_cmpstr(gbinder_client_interface2(client, 11), == ,"11");
g_assert_cmpstr(gbinder_client_interface2(client, 22), == ,"22");
g_assert_cmpstr(gbinder_client_interface2(client, 33), == ,"33");
g_assert(!gbinder_client_interface2(client, 34));
g_assert(!gbinder_client_new_request2(client, 34));
gbinder_client_unref(client);
/* Client with no interface info */
client = gbinder_client_new2(obj, NULL, 0);
g_assert(client);
g_assert(!gbinder_client_interface(client));
g_assert(!gbinder_client_interface2(client, 1));
gbinder_client_unref(client);
gbinder_remote_object_unref(obj);
gbinder_ipc_unref(ipc);
}
/*==========================================================================*
* no_header
*==========================================================================*/
static
void
test_no_header(
void)
{
GBinderClient* client = test_client_new(0, NULL);
int fd = gbinder_driver_fd(gbinder_client_ipc(client)->driver);
test_binder_br_transaction_complete(fd);
g_assert(gbinder_client_transact_sync_oneway(client, 0, NULL) ==
GBINDER_STATUS_OK);
gbinder_client_unref(client);
}
/*==========================================================================*
* dead
*==========================================================================*/
static
void
test_dead_done(
GBinderRemoteObject* obj,
void* user_data)
{
GVERBOSE_("");
test_quit_later((GMainLoop*)user_data);
}
static
void
test_dead(
void)
{
const guint handle = 1;
GBinderClient* client = test_client_new(handle, "foo");
GBinderRemoteObject* obj = client->remote;
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
const int fd = gbinder_driver_fd(gbinder_client_ipc(client)->driver);
gbinder_remote_object_add_death_handler(obj, test_dead_done, loop);
test_binder_br_dead_binder(fd, handle);
test_binder_set_looper_enabled(fd, TEST_LOOPER_ENABLE);
test_run(&test_opt, loop);
g_assert(gbinder_remote_object_is_dead(obj));
g_assert(!gbinder_client_transact_sync_reply(client, 0, NULL, NULL));
g_assert(gbinder_client_transact_sync_oneway(client, 0, NULL) == -ESTALE);
g_assert(!gbinder_client_transact(client, 0, 0, NULL, NULL, NULL, NULL));
gbinder_client_unref(client);
gbinder_ipc_exit();
test_binder_exit_wait(&test_opt, loop);
g_main_loop_unref(loop);
}
/*==========================================================================*
* sync_oneway
*==========================================================================*/
static
void
test_sync_oneway(
void)
{
GBinderClient* client = test_client_new(0, "foo");
GBinderLocalRequest* req = gbinder_client_new_request(client);
int fd = gbinder_driver_fd(gbinder_client_ipc(client)->driver);
g_assert(req);
test_binder_br_transaction_complete(fd);
g_assert(gbinder_client_transact_sync_oneway(client, 0, req) ==
GBINDER_STATUS_OK);
gbinder_local_request_unref(req);
/* Same but using the internal (empty) request */
test_binder_br_transaction_complete(fd);
g_assert(gbinder_client_transact_sync_oneway(client, 0, NULL) ==
GBINDER_STATUS_OK);
gbinder_client_unref(client);
}
/*==========================================================================*
* sync_reply
*==========================================================================*/
static
void
test_sync_reply_tx(
GBinderClient* client,
GBinderLocalRequest* req)
{
GBinderDriver* driver = gbinder_client_ipc(client)->driver;
int fd = gbinder_driver_fd(driver);
const GBinderIo* io = gbinder_driver_io(driver);
GBinderLocalReply* reply = gbinder_local_reply_new(io);
GBinderRemoteReply* tx_reply;
GBinderOutputData* data;
const guint32 handle = 0;
const guint32 code = 1;
const char* result_in = "foo";
char* result_out;
int status = INT_MAX;
g_assert(gbinder_local_reply_append_string16(reply, result_in));
data = gbinder_local_reply_data(reply);
g_assert(data);
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);
g_assert(status == GBINDER_STATUS_OK);
result_out = gbinder_remote_reply_read_string16(tx_reply);
g_assert(!g_strcmp0(result_out, result_in));
g_free(result_out);
gbinder_remote_reply_unref(tx_reply);
gbinder_local_reply_unref(reply);
}
static
void
test_sync_reply(
void)
{
GBinderClient* client = test_client_new(0, "foo");
GBinderLocalRequest* req = gbinder_client_new_request(client);
test_sync_reply_tx(client, req);
gbinder_local_request_unref(req);
/* Same but using the internal (empty) request */
test_sync_reply_tx(client, NULL);
gbinder_client_unref(client);
}
/*==========================================================================*
* reply
*==========================================================================*/
#define TEST_INTERFACE "foo"
#define TEST_REQ_PARAM_STR "bar"
static
void
test_reply_destroy(
void* user_data)
{
test_quit_later((GMainLoop*)user_data);
}
static
void
test_reply_ok_reply(
GBinderClient* client,
GBinderRemoteReply* reply,
int status,
void* user_data)
{
char* result;
GVERBOSE_("%d", status);
g_assert(status == GBINDER_STATUS_OK);
g_assert(reply);
result = gbinder_remote_reply_read_string16(reply);
g_assert(!g_strcmp0(result, TEST_REQ_PARAM_STR));
g_free(result);
}
static
void
test_reply_ok_quit(
GBinderClient* client,
GBinderRemoteReply* reply,
int status,
void* user_data)
{
test_reply_ok_reply(client, reply, status, user_data);
test_reply_destroy(user_data);
}
static
void
test_reply_tx(
GBinderClient* client,
GBinderLocalRequest* req,
GBinderClientReplyFunc done,
GDestroyNotify destroy)
{
GBinderDriver* driver = gbinder_client_ipc(client)->driver;
int fd = gbinder_driver_fd(driver);
const GBinderIo* io = gbinder_driver_io(driver);
GBinderLocalReply* reply = gbinder_local_reply_new(io);
GBinderOutputData* data;
const guint32 handle = 0;
const guint32 code = 1;
GMainLoop* loop = g_main_loop_new(NULL, FALSE);
gulong id;
g_assert(gbinder_local_reply_append_string16(reply, TEST_REQ_PARAM_STR));
data = gbinder_local_reply_data(reply);
g_assert(data);
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);
test_run(&test_opt, loop);
gbinder_local_reply_unref(reply);
g_main_loop_unref(loop);
}
static
void
test_reply(
GBinderClientReplyFunc done,
GDestroyNotify destroy)
{
GBinderClient* client = test_client_new(0, TEST_INTERFACE);
GBinderLocalRequest* req = gbinder_client_new_request2(client, 0);
g_assert(req);
test_reply_tx(client, req, done, destroy);
gbinder_local_request_unref(req);
/* Same but using the internal (empty) request */
test_reply_tx(client, NULL, done, destroy);
gbinder_client_unref(client);
}
static
void
test_reply_ok1(
void)
{
test_reply(test_reply_ok_reply, test_reply_destroy);
}
static
void
test_reply_ok2(
void)
{
test_reply(NULL, test_reply_destroy);
}
static
void
test_reply_ok3(
void)
{
test_reply(test_reply_ok_quit, NULL);
}
/*==========================================================================*
* Common
*==========================================================================*/
#define TEST_PREFIX "/client/"
#define TEST_(t) TEST_PREFIX t
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_("interfaces"), test_interfaces);
g_test_add_func(TEST_("dead"), test_dead);
g_test_add_func(TEST_("no_header"), test_no_header);
g_test_add_func(TEST_("sync_oneway"), test_sync_oneway);
g_test_add_func(TEST_("sync_reply"), test_sync_reply);
g_test_add_func(TEST_("reply/ok1"), test_reply_ok1);
g_test_add_func(TEST_("reply/ok2"), test_reply_ok2);
g_test_add_func(TEST_("reply/ok3"), test_reply_ok3);
test_init(&test_opt, argc, argv);
return g_test_run();
}
/*
* Local Variables:
* mode: C
* c-basic-offset: 4
* indent-tabs-mode: nil
* End:
*/