blob: 89b73a87b1d56bf5b0b843d6beb026776d3513cf [file] [log] [blame]
// Copyright 2016 The Fuchsia Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include <string.h>
#include <magenta/syscalls.h>
#include <magenta/syscalls/object.h>
#include <unittest/unittest.h>
static void check_signals_state(mx_handle_t h, mx_signals_t satisfied) {
mx_signals_t pending = 0;
EXPECT_EQ(mx_object_wait_one(h, 0u, 0u, &pending), MX_ERR_TIMED_OUT, "wrong wait result");
EXPECT_EQ(pending, satisfied, "wrong satisfied state");
}
static bool create_test(void) {
BEGIN_TEST;
{
mx_handle_t h[2] = {MX_HANDLE_INVALID, MX_HANDLE_INVALID};
ASSERT_EQ(mx_eventpair_create(0, &h[0], &h[1]), MX_OK, "eventpair_create failed");
ASSERT_NE(h[0], MX_HANDLE_INVALID, "invalid handle from eventpair_create");
ASSERT_NE(h[1], MX_HANDLE_INVALID, "invalid handle from eventpair_create");
mx_info_handle_basic_t info;
memset(&info, 0, sizeof(info));
mx_status_t status = mx_object_get_info(h[0], MX_INFO_HANDLE_BASIC, &info, sizeof(info), NULL, NULL);
ASSERT_EQ(status, MX_OK, "");
EXPECT_EQ(info.rights,
MX_RIGHT_DUPLICATE | MX_RIGHT_TRANSFER | MX_RIGHT_READ |
MX_RIGHT_WRITE | MX_RIGHT_SIGNAL | MX_RIGHT_SIGNAL_PEER,
"wrong rights");
EXPECT_EQ(info.type, (uint32_t)MX_OBJ_TYPE_EVENT_PAIR, "wrong type");
memset(&info, 0, sizeof(info));
status = mx_object_get_info(h[1], MX_INFO_HANDLE_BASIC, &info, sizeof(info), NULL, NULL);
ASSERT_EQ(status, MX_OK, "");
EXPECT_EQ(info.rights,
MX_RIGHT_DUPLICATE | MX_RIGHT_TRANSFER | MX_RIGHT_READ |
MX_RIGHT_WRITE | MX_RIGHT_SIGNAL | MX_RIGHT_SIGNAL_PEER,
"wrong rights");
EXPECT_EQ(info.type, (uint32_t)MX_OBJ_TYPE_EVENT_PAIR, "wrong type");
EXPECT_EQ(mx_handle_close(h[0]), MX_OK, "failed to close event pair handle");
EXPECT_EQ(mx_handle_close(h[1]), MX_OK, "failed to close event pair handle");
}
// Currently no flags are supported.
{
mx_handle_t h[2] = {MX_HANDLE_INVALID, MX_HANDLE_INVALID};
EXPECT_EQ(mx_eventpair_create(1u, &h[0], &h[1]), MX_ERR_NOT_SUPPORTED, "eventpair_create failed to fail");
EXPECT_EQ(h[0], MX_HANDLE_INVALID, "valid handle from failed eventpair_create?");
EXPECT_EQ(h[1], MX_HANDLE_INVALID, "valid handle from failed eventpair_create?");
}
END_TEST;
}
static bool signal_test(void) {
BEGIN_TEST;
mx_handle_t h[2] = {MX_HANDLE_INVALID, MX_HANDLE_INVALID};
ASSERT_EQ(mx_eventpair_create(0, &h[0], &h[1]), MX_OK, "eventpair_create failed");
ASSERT_NE(h[0], MX_HANDLE_INVALID, "invalid handle from eventpair_create");
ASSERT_NE(h[1], MX_HANDLE_INVALID, "invalid handle from eventpair_create");
check_signals_state(h[0], MX_SIGNAL_LAST_HANDLE);
check_signals_state(h[1], MX_SIGNAL_LAST_HANDLE);
EXPECT_EQ(mx_object_signal(h[0], 0u, MX_USER_SIGNAL_0), MX_OK, "object_signal failed");
check_signals_state(h[1], MX_SIGNAL_LAST_HANDLE);
check_signals_state(h[0], MX_USER_SIGNAL_0 | MX_SIGNAL_LAST_HANDLE);
EXPECT_EQ(mx_object_signal(h[0], MX_USER_SIGNAL_0, 0u), MX_OK, "object_signal failed");
check_signals_state(h[1], MX_SIGNAL_LAST_HANDLE);
check_signals_state(h[0], MX_SIGNAL_LAST_HANDLE);
EXPECT_EQ(mx_handle_close(h[0]), MX_OK, "failed to close event pair handle");
check_signals_state(h[1], MX_EPAIR_PEER_CLOSED | MX_SIGNAL_LAST_HANDLE);
EXPECT_EQ(mx_handle_close(h[1]), MX_OK, "failed to close event pair handle");
END_TEST;
}
static bool signal_peer_test(void) {
BEGIN_TEST;
mx_handle_t h[2] = {MX_HANDLE_INVALID, MX_HANDLE_INVALID};
ASSERT_EQ(mx_eventpair_create(0, &h[0], &h[1]), MX_OK, "eventpair_create failed");
ASSERT_NE(h[0], MX_HANDLE_INVALID, "invalid handle from eventpair_create");
ASSERT_NE(h[1], MX_HANDLE_INVALID, "invalid handle from eventpair_create");
EXPECT_EQ(mx_object_signal_peer(h[0], 0u, MX_USER_SIGNAL_0), MX_OK, "object_signal failed");
check_signals_state(h[0], MX_SIGNAL_LAST_HANDLE);
check_signals_state(h[1], MX_USER_SIGNAL_0 | MX_SIGNAL_LAST_HANDLE);
EXPECT_EQ(mx_object_signal_peer(h[1], 0u, MX_USER_SIGNAL_1 | MX_USER_SIGNAL_2), MX_OK,
"object_signal failed");
check_signals_state(h[0], MX_USER_SIGNAL_1 | MX_USER_SIGNAL_2 | MX_SIGNAL_LAST_HANDLE);
check_signals_state(h[1], MX_USER_SIGNAL_0 | MX_SIGNAL_LAST_HANDLE);
EXPECT_EQ(mx_object_signal_peer(h[0], MX_USER_SIGNAL_0, MX_USER_SIGNAL_3 | MX_USER_SIGNAL_4),
MX_OK, "object_signal failed");
check_signals_state(h[0], MX_USER_SIGNAL_1 | MX_USER_SIGNAL_2 | MX_SIGNAL_LAST_HANDLE);
check_signals_state(h[1], MX_USER_SIGNAL_3 | MX_USER_SIGNAL_4 | MX_SIGNAL_LAST_HANDLE);
EXPECT_EQ(mx_handle_close(h[0]), MX_OK, "failed to close event pair handle");
// Signaled flags should remain satisfied but now should now also get peer closed (and
// unsignaled flags should be unsatisfiable).
check_signals_state(h[1],
MX_EPAIR_PEER_CLOSED | MX_USER_SIGNAL_3 | MX_USER_SIGNAL_4 | MX_SIGNAL_LAST_HANDLE);
EXPECT_EQ(mx_handle_close(h[1]), MX_OK, "failed to close event pair handle");
END_TEST;
}
BEGIN_TEST_CASE(event_pair_tests)
RUN_TEST(create_test)
RUN_TEST(signal_test)
RUN_TEST(signal_peer_test)
END_TEST_CASE(event_pair_tests)
#ifndef BUILD_COMBINED_TESTS
int main(int argc, char** argv) {
return unittest_run_all_tests(argc, argv) ? 0 : -1;
}
#endif