summaryrefslogtreecommitdiffstats
path: root/rlz/lib
diff options
context:
space:
mode:
Diffstat (limited to 'rlz/lib')
-rw-r--r--rlz/lib/rlz_lib_test.cc52
1 files changed, 51 insertions, 1 deletions
diff --git a/rlz/lib/rlz_lib_test.cc b/rlz/lib/rlz_lib_test.cc
index 0f8cb4c..118f48c 100644
--- a/rlz/lib/rlz_lib_test.cc
+++ b/rlz/lib/rlz_lib_test.cc
@@ -13,6 +13,7 @@
// The "GGLA" brand is used to test the normal code flow of the code, and the
// "TEST" brand is used to test the supplementary brand code code flow.
+#include "base/eintr_wrapper.h"
#include "base/logging.h"
#include "base/memory/scoped_ptr.h"
#include "testing/gmock/include/gmock/gmock.h"
@@ -795,7 +796,9 @@ class ReadonlyRlzDirectoryTest : public RlzLibTestNoMachineState {
void ReadonlyRlzDirectoryTest::SetUp() {
RlzLibTestNoMachineState::SetUp();
// Make the rlz directory non-writeable.
- chmod(temp_dir_.path().value().c_str(), 0500);
+ int chmod_result = chmod(temp_dir_.path().value().c_str(), 0500);
+ ASSERT_EQ(0, chmod_result);
+
}
TEST_F(ReadonlyRlzDirectoryTest, WriteFails) {
@@ -821,4 +824,51 @@ TEST_F(ReadonlyRlzDirectoryTest, SupplementaryBrandingDoesNotCrash) {
EXPECT_FALSE(rlz_lib::RecordProductEvent(rlz_lib::TOOLBAR_NOTIFIER,
rlz_lib::IE_DEFAULT_SEARCH, rlz_lib::INSTALL));
}
+
+// Regression test for http://crbug.com/141108
+TEST_F(RlzLibTest, ConcurrentStoreAccessWithProcessExitsWhileLockHeld) {
+ // See the comment at the top of WriteFails.
+ if (!rlz_lib::SupplementaryBranding::GetBrand().empty())
+ return;
+
+ std::vector<pid_t> pids;
+ for (int i = 0; i < 10; ++i) {
+ pid_t pid = fork();
+ ASSERT_NE(-1, pid);
+ if (pid == 0) {
+ // Child.
+ {
+ // SupplementaryBranding is a RAII object for the rlz lock.
+ rlz_lib::SupplementaryBranding branding("TEST");
+
+ // Simulate a crash while holding the lock in some of the children.
+ if (i > 0 && i % 3 == 0)
+ _exit(0);
+
+ // Note: Since this is in a forked child, a failing expectation won't
+ // make the test fail. It does however cause lots of "check failed"
+ // error output. The parent process will then check the exit code
+ // below to make the test fail.
+ bool success = rlz_lib::RecordProductEvent(rlz_lib::TOOLBAR_NOTIFIER,
+ rlz_lib::IE_DEFAULT_SEARCH, rlz_lib::INSTALL);
+ EXPECT_TRUE(success);
+ _exit(success ? 0 : 1);
+ }
+ _exit(0);
+ } else {
+ // Parent.
+ pids.push_back(pid);
+ }
+ }
+
+ int status;
+ for (size_t i = 0; i < pids.size(); ++i) {
+ if (HANDLE_EINTR(waitpid(pids[i], &status, 0)) != -1)
+ EXPECT_TRUE(WIFEXITED(status) && WEXITSTATUS(status) == 0);
+ }
+
+ // No child should have the lock at this point, not even the crashed ones.
+ EXPECT_TRUE(rlz_lib::RecordProductEvent(rlz_lib::TOOLBAR_NOTIFIER,
+ rlz_lib::IE_DEFAULT_SEARCH, rlz_lib::INSTALL));
+}
#endif