diff options
author | keybuk@chromium.org <keybuk@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2012-11-27 00:23:25 +0000 |
---|---|---|
committer | keybuk@chromium.org <keybuk@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2012-11-27 00:23:25 +0000 |
commit | d7134d405180c9551641be53d27a700cb4e5d1cd (patch) | |
tree | 873a8f342a97aa53677ddd74cbfbe14658192d31 /dbus | |
parent | 4e7100f09f13d874a9b2b93e01be5305ce33e46c (diff) | |
download | chromium_src-d7134d405180c9551641be53d27a700cb4e5d1cd.zip chromium_src-d7134d405180c9551641be53d27a700cb4e5d1cd.tar.gz chromium_src-d7134d405180c9551641be53d27a700cb4e5d1cd.tar.bz2 |
Allow multiple object proxies to handle NameOwnerChanged
Multiple object proxies may be registered for any single D-Bus service, one per interface. Returning "handled" means only the first interface object proxy gets to handle the signal, others never see it.
Change to always return "not yet handled", while this means D-Bus never knows we actually handled the NameOwnerChanged signal, that's quite ok and is actually what all other bindings do for all signals.
Submitted as -r169164 but reverted in -r169170; resubmitted with change to run message loop three times if necessary.
BUG=chromium-os:36486
TEST=dbus_unittests
TBR=gauravsh@chromium.org
Review URL: https://codereview.chromium.org/11280073
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@169538 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'dbus')
-rw-r--r-- | dbus/object_proxy.cc | 4 | ||||
-rw-r--r-- | dbus/signal_sender_verification_unittest.cc | 78 |
2 files changed, 75 insertions, 7 deletions
diff --git a/dbus/object_proxy.cc b/dbus/object_proxy.cc index 87950d1..b20f544 100644 --- a/dbus/object_proxy.cc +++ b/dbus/object_proxy.cc @@ -661,11 +661,11 @@ DBusHandlerResult ObjectProxy::HandleNameOwnerChanged( name_owner_changed_callback_, released_signal)); } - return DBUS_HANDLER_RESULT_HANDLED; } } - // Untrusted or uninteresting signal + // Always return unhandled to let other object proxies handle the same + // signal. return DBUS_HANDLER_RESULT_NOT_YET_HANDLED; } diff --git a/dbus/signal_sender_verification_unittest.cc b/dbus/signal_sender_verification_unittest.cc index 6e59ea4..cc665f2 100644 --- a/dbus/signal_sender_verification_unittest.cc +++ b/dbus/signal_sender_verification_unittest.cc @@ -51,7 +51,8 @@ class SignalSenderVerificationTest : public testing::Test { object_proxy_->SetNameOwnerChangedCallback( base::Bind(&SignalSenderVerificationTest::OnNameOwnerChanged, - base::Unretained(this))); + base::Unretained(this), + &on_name_owner_changed_called_)); // Connect to the "Test" signal of "org.chromium.TestInterface" from // the remote object. @@ -119,19 +120,17 @@ class SignalSenderVerificationTest : public testing::Test { message_loop_.Quit(); } - void OnNameOwnerChanged(dbus::Signal* signal) { + void OnNameOwnerChanged(bool* called_flag, dbus::Signal* signal) { dbus::MessageReader reader(signal); std::string name, old_owner, new_owner; ASSERT_TRUE(reader.PopString(&name)); ASSERT_TRUE(reader.PopString(&old_owner)); ASSERT_TRUE(reader.PopString(&new_owner)); latest_name_owner_ = new_owner; - on_name_owner_changed_called_ = true; + *called_flag = true; message_loop_.Quit(); } - protected: - // Called when the "Test" signal is received, in the main thread. // Copy the string payload to |test_signal_string_|. void OnTestSignal(dbus::Signal* signal) { @@ -148,6 +147,8 @@ class SignalSenderVerificationTest : public testing::Test { message_loop_.Quit(); } + protected: + // Wait for the hey signal to be received. void WaitForTestSignal() { // OnTestSignal() will quit the message loop. @@ -245,3 +246,70 @@ TEST_F(SignalSenderVerificationTest, TestOwnerChanged) { WaitForTestSignal(); ASSERT_EQ(kNewMessage, test_signal_string_); } + +TEST_F(SignalSenderVerificationTest, TestMultipleObjects) { + const char kMessage[] = "hello, world"; + + dbus::ObjectProxy* object_proxy2 = bus_->GetObjectProxy( + "org.chromium.TestService", + dbus::ObjectPath("/org/chromium/DifferentObject")); + + bool second_name_owner_changed_called = false; + object_proxy2->SetNameOwnerChangedCallback( + base::Bind(&SignalSenderVerificationTest::OnNameOwnerChanged, + base::Unretained(this), + &second_name_owner_changed_called)); + + // Connect to a signal on the additional remote object to trigger the + // name owner matching. + object_proxy2->ConnectToSignal( + "org.chromium.DifferentTestInterface", + "Test", + base::Bind(&SignalSenderVerificationTest::OnTestSignal, + base::Unretained(this)), + base::Bind(&SignalSenderVerificationTest::OnConnected, + base::Unretained(this))); + // Wait until the object proxy is connected to the signal. + message_loop_.Run(); + + // Send the test signal from the exported object. + test_service_->SendTestSignal(kMessage); + // Receive the signal with the object proxy. The signal is handled in + // SignalSenderVerificationTest::OnTestSignal() in the main thread. + WaitForTestSignal(); + ASSERT_EQ(kMessage, test_signal_string_); + + // Release and acquire the name ownership. + // latest_name_owner_ should be non empty as |test_service_| owns the name. + ASSERT_FALSE(latest_name_owner_.empty()); + test_service_->ShutdownAndBlock(); + // OnNameOwnerChanged will PostTask to quit the message loop. + message_loop_.Run(); + // latest_name_owner_ should be empty as the owner is gone. + ASSERT_TRUE(latest_name_owner_.empty()); + + // Reset the flag as NameOwnerChanged is already received in setup. + on_name_owner_changed_called_ = false; + second_name_owner_changed_called = false; + test_service2_->RequestOwnership( + base::Bind(&SignalSenderVerificationTest::OnOwnership, + base::Unretained(this), true)); + // Both of OnNameOwnerChanged() and OnOwnership() should quit the MessageLoop, + // but there's no expected order of those 2 event. + while (!on_name_owner_changed_called_ || !second_name_owner_changed_called || + !on_ownership_called_) + message_loop_.Run(); + ASSERT_TRUE(on_name_owner_changed_called_); + ASSERT_TRUE(second_name_owner_changed_called); + ASSERT_TRUE(on_ownership_called_); + + // latest_name_owner_ becomes non empty as the new owner appears. + ASSERT_FALSE(latest_name_owner_.empty()); + + // Now the second service owns the name. + const char kNewMessage[] = "hello, new world"; + + test_service2_->SendTestSignal(kNewMessage); + WaitForTestSignal(); + ASSERT_EQ(kNewMessage, test_signal_string_); +} |