1 // Copyright 2007, Google Inc.
2 // All rights reserved.
3 //
4 // Redistribution and use in source and binary forms, with or without
5 // modification, are permitted provided that the following conditions are
6 // met:
7 //
8 //     * Redistributions of source code must retain the above copyright
9 // notice, this list of conditions and the following disclaimer.
10 //     * Redistributions in binary form must reproduce the above
11 // copyright notice, this list of conditions and the following disclaimer
12 // in the documentation and/or other materials provided with the
13 // distribution.
14 //     * Neither the name of Google Inc. nor the names of its
15 // contributors may be used to endorse or promote products derived from
16 // this software without specific prior written permission.
17 //
18 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29 
30 // Silence C4503 (decorated name length exceeded) for MSVC.
31 #ifdef _MSC_VER
32 #pragma warning(push)
33 #pragma warning(disable : 4503)
34 #endif
35 
36 // Google Mock - a framework for writing C++ mock classes.
37 //
38 // This file tests the function mocker classes.
39 #include "gmock/gmock-function-mocker.h"
40 
41 #if GTEST_OS_WINDOWS
42 // MSDN says the header file to be included for STDMETHOD is BaseTyps.h but
43 // we are getting compiler errors if we use basetyps.h, hence including
44 // objbase.h for definition of STDMETHOD.
45 #include <objbase.h>
46 #endif  // GTEST_OS_WINDOWS
47 
48 #include <functional>
49 #include <map>
50 #include <string>
51 #include <type_traits>
52 
53 #include "gmock/gmock.h"
54 #include "gtest/gtest.h"
55 
56 namespace testing {
57 namespace gmock_function_mocker_test {
58 
59 using testing::_;
60 using testing::A;
61 using testing::An;
62 using testing::AnyNumber;
63 using testing::Const;
64 using testing::DoDefault;
65 using testing::Eq;
66 using testing::Lt;
67 using testing::MockFunction;
68 using testing::Ref;
69 using testing::Return;
70 using testing::ReturnRef;
71 using testing::TypedEq;
72 
73 template <typename T>
74 class TemplatedCopyable {
75  public:
TemplatedCopyable()76   TemplatedCopyable() {}
77 
78   template <typename U>
TemplatedCopyable(const U& other)79   TemplatedCopyable(const U& other) {}  // NOLINT
80 };
81 
82 class FooInterface {
83  public:
~FooInterface()84   virtual ~FooInterface() {}
85 
86   virtual void VoidReturning(int x) = 0;
87 
88   virtual int Nullary() = 0;
89   virtual bool Unary(int x) = 0;
90   virtual long Binary(short x, int y) = 0;                     // NOLINT
91   virtual int Decimal(bool b, char c, short d, int e, long f,  // NOLINT
92                       float g, double h, unsigned i, char* j,
93                       const std::string& k) = 0;
94 
95   virtual bool TakesNonConstReference(int& n) = 0;  // NOLINT
96   virtual std::string TakesConstReference(const int& n) = 0;
97   virtual bool TakesConst(const int x) = 0;
98 
99   virtual int OverloadedOnArgumentNumber() = 0;
100   virtual int OverloadedOnArgumentNumber(int n) = 0;
101 
102   virtual int OverloadedOnArgumentType(int n) = 0;
103   virtual char OverloadedOnArgumentType(char c) = 0;
104 
105   virtual int OverloadedOnConstness() = 0;
106   virtual char OverloadedOnConstness() const = 0;
107 
108   virtual int TypeWithHole(int (*func)()) = 0;
109   virtual int TypeWithComma(const std::map<int, std::string>& a_map) = 0;
110   virtual int TypeWithTemplatedCopyCtor(const TemplatedCopyable<int>&) = 0;
111 
112   virtual int (*ReturnsFunctionPointer1(int))(bool) = 0;
113   using fn_ptr = int (*)(bool);
114   virtual fn_ptr ReturnsFunctionPointer2(int) = 0;
115 
116   virtual int RefQualifiedConstRef() const& = 0;
117   virtual int RefQualifiedConstRefRef() const&& = 0;
118   virtual int RefQualifiedRef() & = 0;
119   virtual int RefQualifiedRefRef() && = 0;
120 
121   virtual int RefQualifiedOverloaded() const& = 0;
122   virtual int RefQualifiedOverloaded() const&& = 0;
123   virtual int RefQualifiedOverloaded() & = 0;
124   virtual int RefQualifiedOverloaded() && = 0;
125 
126 #if GTEST_OS_WINDOWS
127   STDMETHOD_(int, CTNullary)() = 0;
128   STDMETHOD_(bool, CTUnary)(int x) = 0;
129   STDMETHOD_(int, CTDecimal)
130   (bool b, char c, short d, int e, long f,  // NOLINT
131    float g, double h, unsigned i, char* j, const std::string& k) = 0;
132   STDMETHOD_(char, CTConst)(int x) const = 0;
133 #endif  // GTEST_OS_WINDOWS
134 };
135 
136 // Const qualifiers on arguments were once (incorrectly) considered
137 // significant in determining whether two virtual functions had the same
138 // signature. This was fixed in Visual Studio 2008. However, the compiler
139 // still emits a warning that alerts about this change in behavior.
140 #ifdef _MSC_VER
141 #pragma warning(push)
142 #pragma warning(disable : 4373)
143 #endif
144 class MockFoo : public FooInterface {
145  public:
MockFoo()146   MockFoo() {}
147 
148   // Makes sure that a mock function parameter can be named.
149   MOCK_METHOD(void, VoidReturning, (int n));  // NOLINT
150 
151   MOCK_METHOD(int, Nullary, ());  // NOLINT
152 
153   // Makes sure that a mock function parameter can be unnamed.
154   MOCK_METHOD(bool, Unary, (int));          // NOLINT
155   MOCK_METHOD(long, Binary, (short, int));  // NOLINT
156   MOCK_METHOD(int, Decimal,
157               (bool, char, short, int, long, float,  // NOLINT
158                double, unsigned, char*, const std::string& str),
159               (override));
160 
161   MOCK_METHOD(bool, TakesNonConstReference, (int&));  // NOLINT
162   MOCK_METHOD(std::string, TakesConstReference, (const int&));
163   MOCK_METHOD(bool, TakesConst, (const int));  // NOLINT
164 
165   // Tests that the function return type can contain unprotected comma.
166   MOCK_METHOD((std::map<int, std::string>), ReturnTypeWithComma, (), ());
167   MOCK_METHOD((std::map<int, std::string>), ReturnTypeWithComma, (int),
168               (const));  // NOLINT
169 
170   MOCK_METHOD(int, OverloadedOnArgumentNumber, ());     // NOLINT
171   MOCK_METHOD(int, OverloadedOnArgumentNumber, (int));  // NOLINT
172 
173   MOCK_METHOD(int, OverloadedOnArgumentType, (int));    // NOLINT
174   MOCK_METHOD(char, OverloadedOnArgumentType, (char));  // NOLINT
175 
176   MOCK_METHOD(int, OverloadedOnConstness, (), (override));          // NOLINT
177   MOCK_METHOD(char, OverloadedOnConstness, (), (override, const));  // NOLINT
178 
179   MOCK_METHOD(int, TypeWithHole, (int (*)()), ());  // NOLINT
180   MOCK_METHOD(int, TypeWithComma, ((const std::map<int, std::string>&)));
181   MOCK_METHOD(int, TypeWithTemplatedCopyCtor,
182               (const TemplatedCopyable<int>&));  // NOLINT
183 
184   MOCK_METHOD(int (*)(bool), ReturnsFunctionPointer1, (int), ());
185   MOCK_METHOD(fn_ptr, ReturnsFunctionPointer2, (int), ());
186 
187 #if GTEST_OS_WINDOWS
188   MOCK_METHOD(int, CTNullary, (), (Calltype(STDMETHODCALLTYPE)));
189   MOCK_METHOD(bool, CTUnary, (int), (Calltype(STDMETHODCALLTYPE)));
190   MOCK_METHOD(int, CTDecimal,
191               (bool b, char c, short d, int e, long f, float g, double h,
192                unsigned i, char* j, const std::string& k),
193               (Calltype(STDMETHODCALLTYPE)));
194   MOCK_METHOD(char, CTConst, (int), (const, Calltype(STDMETHODCALLTYPE)));
195   MOCK_METHOD((std::map<int, std::string>), CTReturnTypeWithComma, (),
196               (Calltype(STDMETHODCALLTYPE)));
197 #endif  // GTEST_OS_WINDOWS
198 
199   // Test reference qualified functions.
200   MOCK_METHOD(int, RefQualifiedConstRef, (), (const, ref(&), override));
201   MOCK_METHOD(int, RefQualifiedConstRefRef, (), (const, ref(&&), override));
202   MOCK_METHOD(int, RefQualifiedRef, (), (ref(&), override));
203   MOCK_METHOD(int, RefQualifiedRefRef, (), (ref(&&), override));
204 
205   MOCK_METHOD(int, RefQualifiedOverloaded, (), (const, ref(&), override));
206   MOCK_METHOD(int, RefQualifiedOverloaded, (), (const, ref(&&), override));
207   MOCK_METHOD(int, RefQualifiedOverloaded, (), (ref(&), override));
208   MOCK_METHOD(int, RefQualifiedOverloaded, (), (ref(&&), override));
209 
210  private:
211   MockFoo(const MockFoo&) = delete;
212   MockFoo& operator=(const MockFoo&) = delete;
213 };
214 
215 class LegacyMockFoo : public FooInterface {
216  public:
LegacyMockFoo()217   LegacyMockFoo() {}
218 
219   // Makes sure that a mock function parameter can be named.
220   MOCK_METHOD1(VoidReturning, void(int n));  // NOLINT
221 
222   MOCK_METHOD0(Nullary, int());  // NOLINT
223 
224   // Makes sure that a mock function parameter can be unnamed.
225   MOCK_METHOD1(Unary, bool(int));                                  // NOLINT
226   MOCK_METHOD2(Binary, long(short, int));                          // NOLINT
227   MOCK_METHOD10(Decimal, int(bool, char, short, int, long, float,  // NOLINT
228                              double, unsigned, char*, const std::string& str));
229 
230   MOCK_METHOD1(TakesNonConstReference, bool(int&));  // NOLINT
231   MOCK_METHOD1(TakesConstReference, std::string(const int&));
232   MOCK_METHOD1(TakesConst, bool(const int));  // NOLINT
233 
234   // Tests that the function return type can contain unprotected comma.
235   MOCK_METHOD0(ReturnTypeWithComma, std::map<int, std::string>());
236   MOCK_CONST_METHOD1(ReturnTypeWithComma,
237                      std::map<int, std::string>(int));  // NOLINT
238 
239   MOCK_METHOD0(OverloadedOnArgumentNumber, int());     // NOLINT
240   MOCK_METHOD1(OverloadedOnArgumentNumber, int(int));  // NOLINT
241 
242   MOCK_METHOD1(OverloadedOnArgumentType, int(int));    // NOLINT
243   MOCK_METHOD1(OverloadedOnArgumentType, char(char));  // NOLINT
244 
245   MOCK_METHOD0(OverloadedOnConstness, int());         // NOLINT
246   MOCK_CONST_METHOD0(OverloadedOnConstness, char());  // NOLINT
247 
248   MOCK_METHOD1(TypeWithHole, int(int (*)()));  // NOLINT
249   MOCK_METHOD1(TypeWithComma,
250                int(const std::map<int, std::string>&));  // NOLINT
251   MOCK_METHOD1(TypeWithTemplatedCopyCtor,
252                int(const TemplatedCopyable<int>&));  // NOLINT
253 
254   MOCK_METHOD1(ReturnsFunctionPointer1, int (*(int))(bool));
255   MOCK_METHOD1(ReturnsFunctionPointer2, fn_ptr(int));
256 
257 #if GTEST_OS_WINDOWS
258   MOCK_METHOD0_WITH_CALLTYPE(STDMETHODCALLTYPE, CTNullary, int());
259   MOCK_METHOD1_WITH_CALLTYPE(STDMETHODCALLTYPE, CTUnary, bool(int));  // NOLINT
260   MOCK_METHOD10_WITH_CALLTYPE(STDMETHODCALLTYPE, CTDecimal,
261                               int(bool b, char c, short d, int e,  // NOLINT
262                                   long f, float g, double h,       // NOLINT
263                                   unsigned i, char* j, const std::string& k));
264   MOCK_CONST_METHOD1_WITH_CALLTYPE(STDMETHODCALLTYPE, CTConst,
265                                    char(int));  // NOLINT
266 
267   // Tests that the function return type can contain unprotected comma.
268   MOCK_METHOD0_WITH_CALLTYPE(STDMETHODCALLTYPE, CTReturnTypeWithComma,
269                              std::map<int, std::string>());
270 #endif  // GTEST_OS_WINDOWS
271 
272   // We can't mock these with the old macros, but we need to define them to make
273   // it concrete.
274   int RefQualifiedConstRef() const& override { return 0; }
275   int RefQualifiedConstRefRef() const&& override { return 0; }
276   int RefQualifiedRef() & override { return 0; }
277   int RefQualifiedRefRef() && override { return 0; }
278   int RefQualifiedOverloaded() const& override { return 0; }
279   int RefQualifiedOverloaded() const&& override { return 0; }
280   int RefQualifiedOverloaded() & override { return 0; }
281   int RefQualifiedOverloaded() && override { return 0; }
282 
283  private:
284   LegacyMockFoo(const LegacyMockFoo&) = delete;
285   LegacyMockFoo& operator=(const LegacyMockFoo&) = delete;
286 };
287 
288 #ifdef _MSC_VER
289 #pragma warning(pop)
290 #endif
291 
292 template <class T>
293 class FunctionMockerTest : public testing::Test {
294  protected:
FunctionMockerTest()295   FunctionMockerTest() : foo_(&mock_foo_) {}
296 
297   FooInterface* const foo_;
298   T mock_foo_;
299 };
300 using FunctionMockerTestTypes = ::testing::Types<MockFoo, LegacyMockFoo>;
301 TYPED_TEST_SUITE(FunctionMockerTest, FunctionMockerTestTypes);
302 
303 // Tests mocking a void-returning function.
TYPED_TEST(FunctionMockerTest, MocksVoidFunction)304 TYPED_TEST(FunctionMockerTest, MocksVoidFunction) {
305   EXPECT_CALL(this->mock_foo_, VoidReturning(Lt(100)));
306   this->foo_->VoidReturning(0);
307 }
308 
309 // Tests mocking a nullary function.
TYPED_TEST(FunctionMockerTest, MocksNullaryFunction)310 TYPED_TEST(FunctionMockerTest, MocksNullaryFunction) {
311   EXPECT_CALL(this->mock_foo_, Nullary())
312       .WillOnce(DoDefault())
313       .WillOnce(Return(1));
314 
315   EXPECT_EQ(0, this->foo_->Nullary());
316   EXPECT_EQ(1, this->foo_->Nullary());
317 }
318 
319 // Tests mocking a unary function.
TYPED_TEST(FunctionMockerTest, MocksUnaryFunction)320 TYPED_TEST(FunctionMockerTest, MocksUnaryFunction) {
321   EXPECT_CALL(this->mock_foo_, Unary(Eq(2))).Times(2).WillOnce(Return(true));
322 
323   EXPECT_TRUE(this->foo_->Unary(2));
324   EXPECT_FALSE(this->foo_->Unary(2));
325 }
326 
327 // Tests mocking a binary function.
TYPED_TEST(FunctionMockerTest, MocksBinaryFunction)328 TYPED_TEST(FunctionMockerTest, MocksBinaryFunction) {
329   EXPECT_CALL(this->mock_foo_, Binary(2, _)).WillOnce(Return(3));
330 
331   EXPECT_EQ(3, this->foo_->Binary(2, 1));
332 }
333 
334 // Tests mocking a decimal function.
TYPED_TEST(FunctionMockerTest, MocksDecimalFunction)335 TYPED_TEST(FunctionMockerTest, MocksDecimalFunction) {
336   EXPECT_CALL(this->mock_foo_,
337               Decimal(true, 'a', 0, 0, 1L, A<float>(), Lt(100), 5U, NULL, "hi"))
338       .WillOnce(Return(5));
339 
340   EXPECT_EQ(5, this->foo_->Decimal(true, 'a', 0, 0, 1, 0, 0, 5, nullptr, "hi"));
341 }
342 
343 // Tests mocking a function that takes a non-const reference.
TYPED_TEST(FunctionMockerTest, MocksFunctionWithNonConstReferenceArgument)344 TYPED_TEST(FunctionMockerTest, MocksFunctionWithNonConstReferenceArgument) {
345   int a = 0;
346   EXPECT_CALL(this->mock_foo_, TakesNonConstReference(Ref(a)))
347       .WillOnce(Return(true));
348 
349   EXPECT_TRUE(this->foo_->TakesNonConstReference(a));
350 }
351 
352 // Tests mocking a function that takes a const reference.
TYPED_TEST(FunctionMockerTest, MocksFunctionWithConstReferenceArgument)353 TYPED_TEST(FunctionMockerTest, MocksFunctionWithConstReferenceArgument) {
354   int a = 0;
355   EXPECT_CALL(this->mock_foo_, TakesConstReference(Ref(a)))
356       .WillOnce(Return("Hello"));
357 
358   EXPECT_EQ("Hello", this->foo_->TakesConstReference(a));
359 }
360 
361 // Tests mocking a function that takes a const variable.
TYPED_TEST(FunctionMockerTest, MocksFunctionWithConstArgument)362 TYPED_TEST(FunctionMockerTest, MocksFunctionWithConstArgument) {
363   EXPECT_CALL(this->mock_foo_, TakesConst(Lt(10))).WillOnce(DoDefault());
364 
365   EXPECT_FALSE(this->foo_->TakesConst(5));
366 }
367 
368 // Tests mocking functions overloaded on the number of arguments.
TYPED_TEST(FunctionMockerTest, MocksFunctionsOverloadedOnArgumentNumber)369 TYPED_TEST(FunctionMockerTest, MocksFunctionsOverloadedOnArgumentNumber) {
370   EXPECT_CALL(this->mock_foo_, OverloadedOnArgumentNumber())
371       .WillOnce(Return(1));
372   EXPECT_CALL(this->mock_foo_, OverloadedOnArgumentNumber(_))
373       .WillOnce(Return(2));
374 
375   EXPECT_EQ(2, this->foo_->OverloadedOnArgumentNumber(1));
376   EXPECT_EQ(1, this->foo_->OverloadedOnArgumentNumber());
377 }
378 
379 // Tests mocking functions overloaded on the types of argument.
TYPED_TEST(FunctionMockerTest, MocksFunctionsOverloadedOnArgumentType)380 TYPED_TEST(FunctionMockerTest, MocksFunctionsOverloadedOnArgumentType) {
381   EXPECT_CALL(this->mock_foo_, OverloadedOnArgumentType(An<int>()))
382       .WillOnce(Return(1));
383   EXPECT_CALL(this->mock_foo_, OverloadedOnArgumentType(TypedEq<char>('a')))
384       .WillOnce(Return('b'));
385 
386   EXPECT_EQ(1, this->foo_->OverloadedOnArgumentType(0));
387   EXPECT_EQ('b', this->foo_->OverloadedOnArgumentType('a'));
388 }
389 
390 // Tests mocking functions overloaded on the const-ness of this object.
TYPED_TEST(FunctionMockerTest, MocksFunctionsOverloadedOnConstnessOfThis)391 TYPED_TEST(FunctionMockerTest, MocksFunctionsOverloadedOnConstnessOfThis) {
392   EXPECT_CALL(this->mock_foo_, OverloadedOnConstness());
393   EXPECT_CALL(Const(this->mock_foo_), OverloadedOnConstness())
394       .WillOnce(Return('a'));
395 
396   EXPECT_EQ(0, this->foo_->OverloadedOnConstness());
397   EXPECT_EQ('a', Const(*this->foo_).OverloadedOnConstness());
398 }
399 
TYPED_TEST(FunctionMockerTest, MocksReturnTypeWithComma)400 TYPED_TEST(FunctionMockerTest, MocksReturnTypeWithComma) {
401   const std::map<int, std::string> a_map;
402   EXPECT_CALL(this->mock_foo_, ReturnTypeWithComma()).WillOnce(Return(a_map));
403   EXPECT_CALL(this->mock_foo_, ReturnTypeWithComma(42)).WillOnce(Return(a_map));
404 
405   EXPECT_EQ(a_map, this->mock_foo_.ReturnTypeWithComma());
406   EXPECT_EQ(a_map, this->mock_foo_.ReturnTypeWithComma(42));
407 }
408 
TYPED_TEST(FunctionMockerTest, MocksTypeWithTemplatedCopyCtor)409 TYPED_TEST(FunctionMockerTest, MocksTypeWithTemplatedCopyCtor) {
410   EXPECT_CALL(this->mock_foo_, TypeWithTemplatedCopyCtor(_))
411       .WillOnce(Return(true));
412   EXPECT_TRUE(this->foo_->TypeWithTemplatedCopyCtor(TemplatedCopyable<int>()));
413 }
414 
415 #if GTEST_OS_WINDOWS
416 // Tests mocking a nullary function with calltype.
TYPED_TEST(FunctionMockerTest, MocksNullaryFunctionWithCallType)417 TYPED_TEST(FunctionMockerTest, MocksNullaryFunctionWithCallType) {
418   EXPECT_CALL(this->mock_foo_, CTNullary())
419       .WillOnce(Return(-1))
420       .WillOnce(Return(0));
421 
422   EXPECT_EQ(-1, this->foo_->CTNullary());
423   EXPECT_EQ(0, this->foo_->CTNullary());
424 }
425 
426 // Tests mocking a unary function with calltype.
TYPED_TEST(FunctionMockerTest, MocksUnaryFunctionWithCallType)427 TYPED_TEST(FunctionMockerTest, MocksUnaryFunctionWithCallType) {
428   EXPECT_CALL(this->mock_foo_, CTUnary(Eq(2)))
429       .Times(2)
430       .WillOnce(Return(true))
431       .WillOnce(Return(false));
432 
433   EXPECT_TRUE(this->foo_->CTUnary(2));
434   EXPECT_FALSE(this->foo_->CTUnary(2));
435 }
436 
437 // Tests mocking a decimal function with calltype.
TYPED_TEST(FunctionMockerTest, MocksDecimalFunctionWithCallType)438 TYPED_TEST(FunctionMockerTest, MocksDecimalFunctionWithCallType) {
439   EXPECT_CALL(this->mock_foo_, CTDecimal(true, 'a', 0, 0, 1L, A<float>(),
440                                          Lt(100), 5U, NULL, "hi"))
441       .WillOnce(Return(10));
442 
443   EXPECT_EQ(10, this->foo_->CTDecimal(true, 'a', 0, 0, 1, 0, 0, 5, NULL, "hi"));
444 }
445 
446 // Tests mocking functions overloaded on the const-ness of this object.
TYPED_TEST(FunctionMockerTest, MocksFunctionsConstFunctionWithCallType)447 TYPED_TEST(FunctionMockerTest, MocksFunctionsConstFunctionWithCallType) {
448   EXPECT_CALL(Const(this->mock_foo_), CTConst(_)).WillOnce(Return('a'));
449 
450   EXPECT_EQ('a', Const(*this->foo_).CTConst(0));
451 }
452 
TYPED_TEST(FunctionMockerTest, MocksReturnTypeWithCommaAndCallType)453 TYPED_TEST(FunctionMockerTest, MocksReturnTypeWithCommaAndCallType) {
454   const std::map<int, std::string> a_map;
455   EXPECT_CALL(this->mock_foo_, CTReturnTypeWithComma()).WillOnce(Return(a_map));
456 
457   EXPECT_EQ(a_map, this->mock_foo_.CTReturnTypeWithComma());
458 }
459 
460 #endif  // GTEST_OS_WINDOWS
461 
TEST(FunctionMockerTest, RefQualified)462 TEST(FunctionMockerTest, RefQualified) {
463   MockFoo mock_foo;
464 
465   EXPECT_CALL(mock_foo, RefQualifiedConstRef).WillOnce(Return(1));
466   EXPECT_CALL(std::move(mock_foo),  // NOLINT
467               RefQualifiedConstRefRef)
468       .WillOnce(Return(2));
469   EXPECT_CALL(mock_foo, RefQualifiedRef).WillOnce(Return(3));
470   EXPECT_CALL(std::move(mock_foo),  // NOLINT
471               RefQualifiedRefRef)
472       .WillOnce(Return(4));
473 
474   EXPECT_CALL(static_cast<const MockFoo&>(mock_foo), RefQualifiedOverloaded())
475       .WillOnce(Return(5));
476   EXPECT_CALL(static_cast<const MockFoo&&>(mock_foo), RefQualifiedOverloaded())
477       .WillOnce(Return(6));
478   EXPECT_CALL(static_cast<MockFoo&>(mock_foo), RefQualifiedOverloaded())
479       .WillOnce(Return(7));
480   EXPECT_CALL(static_cast<MockFoo&&>(mock_foo), RefQualifiedOverloaded())
481       .WillOnce(Return(8));
482 
483   EXPECT_EQ(mock_foo.RefQualifiedConstRef(), 1);
484   EXPECT_EQ(std::move(mock_foo).RefQualifiedConstRefRef(), 2);  // NOLINT
485   EXPECT_EQ(mock_foo.RefQualifiedRef(), 3);
486   EXPECT_EQ(std::move(mock_foo).RefQualifiedRefRef(), 4);  // NOLINT
487 
488   EXPECT_EQ(std::cref(mock_foo).get().RefQualifiedOverloaded(), 5);
489   EXPECT_EQ(std::move(std::cref(mock_foo).get())  // NOLINT
490                 .RefQualifiedOverloaded(),
491             6);
492   EXPECT_EQ(mock_foo.RefQualifiedOverloaded(), 7);
493   EXPECT_EQ(std::move(mock_foo).RefQualifiedOverloaded(), 8);  // NOLINT
494 }
495 
496 class MockB {
497  public:
MockB()498   MockB() {}
499 
500   MOCK_METHOD(void, DoB, ());
501 
502  private:
503   MockB(const MockB&) = delete;
504   MockB& operator=(const MockB&) = delete;
505 };
506 
507 class LegacyMockB {
508  public:
LegacyMockB()509   LegacyMockB() {}
510 
511   MOCK_METHOD0(DoB, void());
512 
513  private:
514   LegacyMockB(const LegacyMockB&) = delete;
515   LegacyMockB& operator=(const LegacyMockB&) = delete;
516 };
517 
518 template <typename T>
519 class ExpectCallTest : public ::testing::Test {};
520 using ExpectCallTestTypes = ::testing::Types<MockB, LegacyMockB>;
521 TYPED_TEST_SUITE(ExpectCallTest, ExpectCallTestTypes);
522 
523 // Tests that functions with no EXPECT_CALL() rules can be called any
524 // number of times.
TYPED_TEST(ExpectCallTest, UnmentionedFunctionCanBeCalledAnyNumberOfTimes)525 TYPED_TEST(ExpectCallTest, UnmentionedFunctionCanBeCalledAnyNumberOfTimes) {
526   { TypeParam b; }
527 
528   {
529     TypeParam b;
530     b.DoB();
531   }
532 
533   {
534     TypeParam b;
535     b.DoB();
536     b.DoB();
537   }
538 }
539 
540 // Tests mocking template interfaces.
541 
542 template <typename T>
543 class StackInterface {
544  public:
~StackInterface()545   virtual ~StackInterface() {}
546 
547   // Template parameter appears in function parameter.
548   virtual void Push(const T& value) = 0;
549   virtual void Pop() = 0;
550   virtual int GetSize() const = 0;
551   // Template parameter appears in function return type.
552   virtual const T& GetTop() const = 0;
553 };
554 
555 template <typename T>
556 class MockStack : public StackInterface<T> {
557  public:
MockStack()558   MockStack() {}
559 
560   MOCK_METHOD(void, Push, (const T& elem), ());
561   MOCK_METHOD(void, Pop, (), (final));
562   MOCK_METHOD(int, GetSize, (), (const, override));
563   MOCK_METHOD(const T&, GetTop, (), (const));
564 
565   // Tests that the function return type can contain unprotected comma.
566   MOCK_METHOD((std::map<int, int>), ReturnTypeWithComma, (), ());
567   MOCK_METHOD((std::map<int, int>), ReturnTypeWithComma, (int), (const));
568 
569  private:
570   MockStack(const MockStack&) = delete;
571   MockStack& operator=(const MockStack&) = delete;
572 };
573 
574 template <typename T>
575 class LegacyMockStack : public StackInterface<T> {
576  public:
LegacyMockStack()577   LegacyMockStack() {}
578 
579   MOCK_METHOD1_T(Push, void(const T& elem));
580   MOCK_METHOD0_T(Pop, void());
581   MOCK_CONST_METHOD0_T(GetSize, int());  // NOLINT
582   MOCK_CONST_METHOD0_T(GetTop, const T&());
583 
584   // Tests that the function return type can contain unprotected comma.
585   MOCK_METHOD0_T(ReturnTypeWithComma, std::map<int, int>());
586   MOCK_CONST_METHOD1_T(ReturnTypeWithComma, std::map<int, int>(int));  // NOLINT
587 
588  private:
589   LegacyMockStack(const LegacyMockStack&) = delete;
590   LegacyMockStack& operator=(const LegacyMockStack&) = delete;
591 };
592 
593 template <typename T>
594 class TemplateMockTest : public ::testing::Test {};
595 using TemplateMockTestTypes =
596     ::testing::Types<MockStack<int>, LegacyMockStack<int>>;
597 TYPED_TEST_SUITE(TemplateMockTest, TemplateMockTestTypes);
598 
599 // Tests that template mock works.
TYPED_TEST(TemplateMockTest, Works)600 TYPED_TEST(TemplateMockTest, Works) {
601   TypeParam mock;
602 
603   EXPECT_CALL(mock, GetSize())
604       .WillOnce(Return(0))
605       .WillOnce(Return(1))
606       .WillOnce(Return(0));
607   EXPECT_CALL(mock, Push(_));
608   int n = 5;
609   EXPECT_CALL(mock, GetTop()).WillOnce(ReturnRef(n));
610   EXPECT_CALL(mock, Pop()).Times(AnyNumber());
611 
612   EXPECT_EQ(0, mock.GetSize());
613   mock.Push(5);
614   EXPECT_EQ(1, mock.GetSize());
615   EXPECT_EQ(5, mock.GetTop());
616   mock.Pop();
617   EXPECT_EQ(0, mock.GetSize());
618 }
619 
TYPED_TEST(TemplateMockTest, MethodWithCommaInReturnTypeWorks)620 TYPED_TEST(TemplateMockTest, MethodWithCommaInReturnTypeWorks) {
621   TypeParam mock;
622 
623   const std::map<int, int> a_map;
624   EXPECT_CALL(mock, ReturnTypeWithComma()).WillOnce(Return(a_map));
625   EXPECT_CALL(mock, ReturnTypeWithComma(1)).WillOnce(Return(a_map));
626 
627   EXPECT_EQ(a_map, mock.ReturnTypeWithComma());
628   EXPECT_EQ(a_map, mock.ReturnTypeWithComma(1));
629 }
630 
631 #if GTEST_OS_WINDOWS
632 // Tests mocking template interfaces with calltype.
633 
634 template <typename T>
635 class StackInterfaceWithCallType {
636  public:
~StackInterfaceWithCallType()637   virtual ~StackInterfaceWithCallType() {}
638 
639   // Template parameter appears in function parameter.
640   STDMETHOD_(void, Push)(const T& value) = 0;
641   STDMETHOD_(void, Pop)() = 0;
642   STDMETHOD_(int, GetSize)() const = 0;
643   // Template parameter appears in function return type.
644   STDMETHOD_(const T&, GetTop)() const = 0;
645 };
646 
647 template <typename T>
648 class MockStackWithCallType : public StackInterfaceWithCallType<T> {
649  public:
MockStackWithCallType()650   MockStackWithCallType() {}
651 
652   MOCK_METHOD(void, Push, (const T& elem),
653               (Calltype(STDMETHODCALLTYPE), override));
654   MOCK_METHOD(void, Pop, (), (Calltype(STDMETHODCALLTYPE), override));
655   MOCK_METHOD(int, GetSize, (), (Calltype(STDMETHODCALLTYPE), override, const));
656   MOCK_METHOD(const T&, GetTop, (),
657               (Calltype(STDMETHODCALLTYPE), override, const));
658 
659  private:
660   MockStackWithCallType(const MockStackWithCallType&) = delete;
661   MockStackWithCallType& operator=(const MockStackWithCallType&) = delete;
662 };
663 
664 template <typename T>
665 class LegacyMockStackWithCallType : public StackInterfaceWithCallType<T> {
666  public:
LegacyMockStackWithCallType()667   LegacyMockStackWithCallType() {}
668 
669   MOCK_METHOD1_T_WITH_CALLTYPE(STDMETHODCALLTYPE, Push, void(const T& elem));
670   MOCK_METHOD0_T_WITH_CALLTYPE(STDMETHODCALLTYPE, Pop, void());
671   MOCK_CONST_METHOD0_T_WITH_CALLTYPE(STDMETHODCALLTYPE, GetSize, int());
672   MOCK_CONST_METHOD0_T_WITH_CALLTYPE(STDMETHODCALLTYPE, GetTop, const T&());
673 
674  private:
675   LegacyMockStackWithCallType(const LegacyMockStackWithCallType&) = delete;
676   LegacyMockStackWithCallType& operator=(const LegacyMockStackWithCallType&) =
677       delete;
678 };
679 
680 template <typename T>
681 class TemplateMockTestWithCallType : public ::testing::Test {};
682 using TemplateMockTestWithCallTypeTypes =
683     ::testing::Types<MockStackWithCallType<int>,
684                      LegacyMockStackWithCallType<int>>;
685 TYPED_TEST_SUITE(TemplateMockTestWithCallType,
686                  TemplateMockTestWithCallTypeTypes);
687 
688 // Tests that template mock with calltype works.
TYPED_TEST(TemplateMockTestWithCallType, Works)689 TYPED_TEST(TemplateMockTestWithCallType, Works) {
690   TypeParam mock;
691 
692   EXPECT_CALL(mock, GetSize())
693       .WillOnce(Return(0))
694       .WillOnce(Return(1))
695       .WillOnce(Return(0));
696   EXPECT_CALL(mock, Push(_));
697   int n = 5;
698   EXPECT_CALL(mock, GetTop()).WillOnce(ReturnRef(n));
699   EXPECT_CALL(mock, Pop()).Times(AnyNumber());
700 
701   EXPECT_EQ(0, mock.GetSize());
702   mock.Push(5);
703   EXPECT_EQ(1, mock.GetSize());
704   EXPECT_EQ(5, mock.GetTop());
705   mock.Pop();
706   EXPECT_EQ(0, mock.GetSize());
707 }
708 #endif  // GTEST_OS_WINDOWS
709 
710 #define MY_MOCK_METHODS1_                       \
711   MOCK_METHOD(void, Overloaded, ());            \
712   MOCK_METHOD(int, Overloaded, (int), (const)); \
713   MOCK_METHOD(bool, Overloaded, (bool f, int n))
714 
715 #define LEGACY_MY_MOCK_METHODS1_              \
716   MOCK_METHOD0(Overloaded, void());           \
717   MOCK_CONST_METHOD1(Overloaded, int(int n)); \
718   MOCK_METHOD2(Overloaded, bool(bool f, int n))
719 
720 class MockOverloadedOnArgNumber {
721  public:
MockOverloadedOnArgNumber()722   MockOverloadedOnArgNumber() {}
723 
724   MY_MOCK_METHODS1_;
725 
726  private:
727   MockOverloadedOnArgNumber(const MockOverloadedOnArgNumber&) = delete;
728   MockOverloadedOnArgNumber& operator=(const MockOverloadedOnArgNumber&) =
729       delete;
730 };
731 
732 class LegacyMockOverloadedOnArgNumber {
733  public:
LegacyMockOverloadedOnArgNumber()734   LegacyMockOverloadedOnArgNumber() {}
735 
736   LEGACY_MY_MOCK_METHODS1_;
737 
738  private:
739   LegacyMockOverloadedOnArgNumber(const LegacyMockOverloadedOnArgNumber&) =
740       delete;
741   LegacyMockOverloadedOnArgNumber& operator=(
742       const LegacyMockOverloadedOnArgNumber&) = delete;
743 };
744 
745 template <typename T>
746 class OverloadedMockMethodTest : public ::testing::Test {};
747 using OverloadedMockMethodTestTypes =
748     ::testing::Types<MockOverloadedOnArgNumber,
749                      LegacyMockOverloadedOnArgNumber>;
750 TYPED_TEST_SUITE(OverloadedMockMethodTest, OverloadedMockMethodTestTypes);
751 
TYPED_TEST(OverloadedMockMethodTest, CanOverloadOnArgNumberInMacroBody)752 TYPED_TEST(OverloadedMockMethodTest, CanOverloadOnArgNumberInMacroBody) {
753   TypeParam mock;
754   EXPECT_CALL(mock, Overloaded());
755   EXPECT_CALL(mock, Overloaded(1)).WillOnce(Return(2));
756   EXPECT_CALL(mock, Overloaded(true, 1)).WillOnce(Return(true));
757 
758   mock.Overloaded();
759   EXPECT_EQ(2, mock.Overloaded(1));
760   EXPECT_TRUE(mock.Overloaded(true, 1));
761 }
762 
763 #define MY_MOCK_METHODS2_                     \
764   MOCK_CONST_METHOD1(Overloaded, int(int n)); \
765   MOCK_METHOD1(Overloaded, int(int n))
766 
767 class MockOverloadedOnConstness {
768  public:
MockOverloadedOnConstness()769   MockOverloadedOnConstness() {}
770 
771   MY_MOCK_METHODS2_;
772 
773  private:
774   MockOverloadedOnConstness(const MockOverloadedOnConstness&) = delete;
775   MockOverloadedOnConstness& operator=(const MockOverloadedOnConstness&) =
776       delete;
777 };
778 
TEST(MockMethodOverloadedMockMethodTest, CanOverloadOnConstnessInMacroBody)779 TEST(MockMethodOverloadedMockMethodTest, CanOverloadOnConstnessInMacroBody) {
780   MockOverloadedOnConstness mock;
781   const MockOverloadedOnConstness* const_mock = &mock;
782   EXPECT_CALL(mock, Overloaded(1)).WillOnce(Return(2));
783   EXPECT_CALL(*const_mock, Overloaded(1)).WillOnce(Return(3));
784 
785   EXPECT_EQ(2, mock.Overloaded(1));
786   EXPECT_EQ(3, const_mock->Overloaded(1));
787 }
788 
TEST(MockMethodMockFunctionTest, WorksForVoidNullary)789 TEST(MockMethodMockFunctionTest, WorksForVoidNullary) {
790   MockFunction<void()> foo;
791   EXPECT_CALL(foo, Call());
792   foo.Call();
793 }
794 
TEST(MockMethodMockFunctionTest, WorksForNonVoidNullary)795 TEST(MockMethodMockFunctionTest, WorksForNonVoidNullary) {
796   MockFunction<int()> foo;
797   EXPECT_CALL(foo, Call()).WillOnce(Return(1)).WillOnce(Return(2));
798   EXPECT_EQ(1, foo.Call());
799   EXPECT_EQ(2, foo.Call());
800 }
801 
TEST(MockMethodMockFunctionTest, WorksForVoidUnary)802 TEST(MockMethodMockFunctionTest, WorksForVoidUnary) {
803   MockFunction<void(int)> foo;
804   EXPECT_CALL(foo, Call(1));
805   foo.Call(1);
806 }
807 
TEST(MockMethodMockFunctionTest, WorksForNonVoidBinary)808 TEST(MockMethodMockFunctionTest, WorksForNonVoidBinary) {
809   MockFunction<int(bool, int)> foo;
810   EXPECT_CALL(foo, Call(false, 42)).WillOnce(Return(1)).WillOnce(Return(2));
811   EXPECT_CALL(foo, Call(true, Ge(100))).WillOnce(Return(3));
812   EXPECT_EQ(1, foo.Call(false, 42));
813   EXPECT_EQ(2, foo.Call(false, 42));
814   EXPECT_EQ(3, foo.Call(true, 120));
815 }
816 
TEST(MockMethodMockFunctionTest, WorksFor10Arguments)817 TEST(MockMethodMockFunctionTest, WorksFor10Arguments) {
818   MockFunction<int(bool a0, char a1, int a2, int a3, int a4, int a5, int a6,
819                    char a7, int a8, bool a9)>
820       foo;
821   EXPECT_CALL(foo, Call(_, 'a', _, _, _, _, _, _, _, _))
822       .WillOnce(Return(1))
823       .WillOnce(Return(2));
824   EXPECT_EQ(1, foo.Call(false, 'a', 0, 0, 0, 0, 0, 'b', 0, true));
825   EXPECT_EQ(2, foo.Call(true, 'a', 0, 0, 0, 0, 0, 'b', 1, false));
826 }
827 
TEST(MockMethodMockFunctionTest, AsStdFunction)828 TEST(MockMethodMockFunctionTest, AsStdFunction) {
829   MockFunction<int(int)> foo;
830   auto call = [](const std::function<int(int)>& f, int i) { return f(i); };
831   EXPECT_CALL(foo, Call(1)).WillOnce(Return(-1));
832   EXPECT_CALL(foo, Call(2)).WillOnce(Return(-2));
833   EXPECT_EQ(-1, call(foo.AsStdFunction(), 1));
834   EXPECT_EQ(-2, call(foo.AsStdFunction(), 2));
835 }
836 
TEST(MockMethodMockFunctionTest, AsStdFunctionReturnsReference)837 TEST(MockMethodMockFunctionTest, AsStdFunctionReturnsReference) {
838   MockFunction<int&()> foo;
839   int value = 1;
840   EXPECT_CALL(foo, Call()).WillOnce(ReturnRef(value));
841   int& ref = foo.AsStdFunction()();
842   EXPECT_EQ(1, ref);
843   value = 2;
844   EXPECT_EQ(2, ref);
845 }
846 
TEST(MockMethodMockFunctionTest, AsStdFunctionWithReferenceParameter)847 TEST(MockMethodMockFunctionTest, AsStdFunctionWithReferenceParameter) {
848   MockFunction<int(int&)> foo;
849   auto call = [](const std::function<int(int&)>& f, int& i) { return f(i); };
850   int i = 42;
851   EXPECT_CALL(foo, Call(i)).WillOnce(Return(-1));
852   EXPECT_EQ(-1, call(foo.AsStdFunction(), i));
853 }
854 
855 namespace {
856 
857 template <typename Expected, typename F>
IsMockFunctionTemplateArgumentDeducedTo( const internal::MockFunction<F>&)858 static constexpr bool IsMockFunctionTemplateArgumentDeducedTo(
859     const internal::MockFunction<F>&) {
860   return std::is_same<F, Expected>::value;
861 }
862 
863 }  // namespace
864 
865 template <typename F>
866 class MockMethodMockFunctionSignatureTest : public Test {};
867 
868 using MockMethodMockFunctionSignatureTypes =
869     Types<void(), int(), void(int), int(int), int(bool, int),
870           int(bool, char, int, int, int, int, int, char, int, bool)>;
871 TYPED_TEST_SUITE(MockMethodMockFunctionSignatureTest,
872                  MockMethodMockFunctionSignatureTypes);
873 
TYPED_TEST(MockMethodMockFunctionSignatureTest, IsMockFunctionTemplateArgumentDeducedForRawSignature)874 TYPED_TEST(MockMethodMockFunctionSignatureTest,
875            IsMockFunctionTemplateArgumentDeducedForRawSignature) {
876   using Argument = TypeParam;
877   MockFunction<Argument> foo;
878   EXPECT_TRUE(IsMockFunctionTemplateArgumentDeducedTo<TypeParam>(foo));
879 }
880 
TYPED_TEST(MockMethodMockFunctionSignatureTest, IsMockFunctionTemplateArgumentDeducedForStdFunction)881 TYPED_TEST(MockMethodMockFunctionSignatureTest,
882            IsMockFunctionTemplateArgumentDeducedForStdFunction) {
883   using Argument = std::function<TypeParam>;
884   MockFunction<Argument> foo;
885   EXPECT_TRUE(IsMockFunctionTemplateArgumentDeducedTo<TypeParam>(foo));
886 }
887 
TYPED_TEST( MockMethodMockFunctionSignatureTest, IsMockFunctionCallMethodSignatureTheSameForRawSignatureAndStdFunction)888 TYPED_TEST(
889     MockMethodMockFunctionSignatureTest,
890     IsMockFunctionCallMethodSignatureTheSameForRawSignatureAndStdFunction) {
891   using ForRawSignature = decltype(&MockFunction<TypeParam>::Call);
892   using ForStdFunction =
893       decltype(&MockFunction<std::function<TypeParam>>::Call);
894   EXPECT_TRUE((std::is_same<ForRawSignature, ForStdFunction>::value));
895 }
896 
897 template <typename F>
898 struct AlternateCallable {};
899 
TYPED_TEST(MockMethodMockFunctionSignatureTest, IsMockFunctionTemplateArgumentDeducedForAlternateCallable)900 TYPED_TEST(MockMethodMockFunctionSignatureTest,
901            IsMockFunctionTemplateArgumentDeducedForAlternateCallable) {
902   using Argument = AlternateCallable<TypeParam>;
903   MockFunction<Argument> foo;
904   EXPECT_TRUE(IsMockFunctionTemplateArgumentDeducedTo<TypeParam>(foo));
905 }
906 
TYPED_TEST(MockMethodMockFunctionSignatureTest, IsMockFunctionCallMethodSignatureTheSameForAlternateCallable)907 TYPED_TEST(MockMethodMockFunctionSignatureTest,
908            IsMockFunctionCallMethodSignatureTheSameForAlternateCallable) {
909   using ForRawSignature = decltype(&MockFunction<TypeParam>::Call);
910   using ForStdFunction =
911       decltype(&MockFunction<std::function<TypeParam>>::Call);
912   EXPECT_TRUE((std::is_same<ForRawSignature, ForStdFunction>::value));
913 }
914 
915 struct MockMethodSizes0 {
916   MOCK_METHOD(void, func, ());
917 };
918 struct MockMethodSizes1 {
919   MOCK_METHOD(void, func, (int));
920 };
921 struct MockMethodSizes2 {
922   MOCK_METHOD(void, func, (int, int));
923 };
924 struct MockMethodSizes3 {
925   MOCK_METHOD(void, func, (int, int, int));
926 };
927 struct MockMethodSizes4 {
928   MOCK_METHOD(void, func, (int, int, int, int));
929 };
930 
931 struct LegacyMockMethodSizes0 {
932   MOCK_METHOD0(func, void());
933 };
934 struct LegacyMockMethodSizes1 {
935   MOCK_METHOD1(func, void(int));
936 };
937 struct LegacyMockMethodSizes2 {
938   MOCK_METHOD2(func, void(int, int));
939 };
940 struct LegacyMockMethodSizes3 {
941   MOCK_METHOD3(func, void(int, int, int));
942 };
943 struct LegacyMockMethodSizes4 {
944   MOCK_METHOD4(func, void(int, int, int, int));
945 };
946 
TEST(MockMethodMockFunctionTest, MockMethodSizeOverhead)947 TEST(MockMethodMockFunctionTest, MockMethodSizeOverhead) {
948   EXPECT_EQ(sizeof(MockMethodSizes0), sizeof(MockMethodSizes1));
949   EXPECT_EQ(sizeof(MockMethodSizes0), sizeof(MockMethodSizes2));
950   EXPECT_EQ(sizeof(MockMethodSizes0), sizeof(MockMethodSizes3));
951   EXPECT_EQ(sizeof(MockMethodSizes0), sizeof(MockMethodSizes4));
952 
953   EXPECT_EQ(sizeof(LegacyMockMethodSizes0), sizeof(LegacyMockMethodSizes1));
954   EXPECT_EQ(sizeof(LegacyMockMethodSizes0), sizeof(LegacyMockMethodSizes2));
955   EXPECT_EQ(sizeof(LegacyMockMethodSizes0), sizeof(LegacyMockMethodSizes3));
956   EXPECT_EQ(sizeof(LegacyMockMethodSizes0), sizeof(LegacyMockMethodSizes4));
957 
958   EXPECT_EQ(sizeof(LegacyMockMethodSizes0), sizeof(MockMethodSizes0));
959 }
960 
TEST(MockMethodMockFunctionTest, EnsureNoUnusedMemberFunction)961 TEST(MockMethodMockFunctionTest, EnsureNoUnusedMemberFunction) {
962 #ifdef __clang__
963 #pragma clang diagnostic push
964 #pragma clang diagnostic error "-Wunused-member-function"
965 #endif
966   // https://github.com/google/googletest/issues/4052
967   struct Foo {
968     MOCK_METHOD(void, foo, ());
969   };
970   EXPECT_CALL(Foo(), foo()).Times(0);
971 #ifdef __clang__
972 #pragma clang diagnostic pop
973 #endif
974 }
975 
976 void hasTwoParams(int, int);
977 void MaybeThrows();
978 void DoesntThrow() noexcept;
979 struct MockMethodNoexceptSpecifier {
980   MOCK_METHOD(void, func1, (), (noexcept));
981   MOCK_METHOD(void, func2, (), (noexcept(true)));
982   MOCK_METHOD(void, func3, (), (noexcept(false)));
983   MOCK_METHOD(void, func4, (), (noexcept(noexcept(MaybeThrows()))));
984   MOCK_METHOD(void, func5, (), (noexcept(noexcept(DoesntThrow()))));
985   MOCK_METHOD(void, func6, (), (noexcept(noexcept(DoesntThrow())), const));
986   MOCK_METHOD(void, func7, (), (const, noexcept(noexcept(DoesntThrow()))));
987   // Put commas in the noexcept expression
988   MOCK_METHOD(void, func8, (), (noexcept(noexcept(hasTwoParams(1, 2))), const));
989 };
990 
TEST(MockMethodMockFunctionTest, NoexceptSpecifierPreserved)991 TEST(MockMethodMockFunctionTest, NoexceptSpecifierPreserved) {
992   EXPECT_TRUE(noexcept(std::declval<MockMethodNoexceptSpecifier>().func1()));
993   EXPECT_TRUE(noexcept(std::declval<MockMethodNoexceptSpecifier>().func2()));
994   EXPECT_FALSE(noexcept(std::declval<MockMethodNoexceptSpecifier>().func3()));
995   EXPECT_FALSE(noexcept(std::declval<MockMethodNoexceptSpecifier>().func4()));
996   EXPECT_TRUE(noexcept(std::declval<MockMethodNoexceptSpecifier>().func5()));
997   EXPECT_TRUE(noexcept(std::declval<MockMethodNoexceptSpecifier>().func6()));
998   EXPECT_TRUE(noexcept(std::declval<MockMethodNoexceptSpecifier>().func7()));
999   EXPECT_EQ(noexcept(std::declval<MockMethodNoexceptSpecifier>().func8()),
1000             noexcept(hasTwoParams(1, 2)));
1001 }
1002 
1003 }  // namespace gmock_function_mocker_test
1004 }  // namespace testing
1005