1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
* You can obtain one at http://mozilla.org/MPL/2.0/. */
/*
* Test that sync messages preempt other messages in the expected way.
*/
#include "gtest/gtest.h"
#include "mozilla/_ipdltest/IPDLUnitTest.h"
#include "mozilla/_ipdltest/PTestUrgencyChild.h"
#include "mozilla/_ipdltest/PTestUrgencyParent.h"
using namespace mozilla::ipc;
namespace mozilla::_ipdltest {
enum {
kFirstTestBegin = 1,
kFirstTestGotReply,
kSecondTestBegin,
kSecondTestGotReply,
};
class TestUrgencyParent : public PTestUrgencyParent {
NS_INLINE_DECL_REFCOUNTING(TestUrgencyParent, override)
private:
IPCResult RecvTest1(uint32_t* value) final override {
EXPECT_TRUE(SendReply1(value));
EXPECT_EQ(*value, (uint32_t)99) << "unexpected value";
return IPC_OK();
}
IPCResult RecvTest2() final override {
uint32_t value;
inreply_ = true;
EXPECT_TRUE(SendReply2(&value));
inreply_ = false;
EXPECT_EQ(value, (uint32_t)500) << "unexpected value";
return IPC_OK();
}
IPCResult RecvTest3(uint32_t* value) final override {
EXPECT_FALSE(inreply_) << "nested non-urgent on top of urgent message";
*value = 1000;
return IPC_OK();
}
IPCResult RecvFinalTest_Begin() final override { return IPC_OK(); }
~TestUrgencyParent() = default;
bool inreply_ = false;
};
class TestUrgencyChild : public PTestUrgencyChild {
NS_INLINE_DECL_REFCOUNTING(TestUrgencyChild, override)
private:
IPCResult RecvStart() final override {
uint32_t result;
// Send a synchronous message, expect to get an urgent message while
// blocked.
test_ = kFirstTestBegin;
EXPECT_TRUE(SendTest1(&result));
EXPECT_EQ(result, (uint32_t)99) << "wrong value from SendTest1";
EXPECT_EQ(test_, kFirstTestGotReply)
<< "never received first urgent message";
// Initiate the next test by sending an asynchronous message, then becoming
// blocked. This tests that the urgent message is still delivered properly,
// and that the parent does not try to service the sync
test_ = kSecondTestBegin;
EXPECT_TRUE(SendTest2());
EXPECT_TRUE(SendTest3(&result));
EXPECT_EQ(test_, kSecondTestGotReply)
<< "never received second urgent message";
EXPECT_EQ(result, (uint32_t)1000) << "wrong value from SendTest3";
EXPECT_TRUE(SendFinalTest_Begin());
Close();
return IPC_OK();
}
IPCResult RecvReply1(uint32_t* reply) final override {
EXPECT_EQ(test_, kFirstTestBegin) << "wrong test state in RecvReply1";
*reply = 99;
test_ = kFirstTestGotReply;
return IPC_OK();
}
IPCResult RecvReply2(uint32_t* reply) final override {
EXPECT_EQ(test_, kSecondTestBegin) << "wrong test state in RecvReply2";
*reply = 500;
test_ = kSecondTestGotReply;
return IPC_OK();
}
~TestUrgencyChild() = default;
uint32_t test_ = 0;
};
// Only run cross-process because we need to send nested sync messages (this can
// only be done from the main thread).
IPDL_TEST_ON(CROSSPROCESS, TestUrgency) { EXPECT_TRUE(mActor->SendStart()); }
} // namespace mozilla::_ipdltest
|