Skip to content

Commit 1f9edcd

Browse files
Abseil TeamAndy Soffer
Abseil Team
authored and
Andy Soffer
committed
Googletest export
Addressing #2502 Add MOCK_METHOD support for returning function pointers. PiperOrigin-RevId: 275323671
1 parent bbe4b73 commit 1f9edcd

File tree

2 files changed

+32
-4
lines changed

2 files changed

+32
-4
lines changed

googlemock/include/gmock/gmock-function-mocker.h

+25-4
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,13 @@
3939
#include "gmock/gmock-generated-function-mockers.h" // NOLINT
4040
#include "gmock/internal/gmock-pp.h"
4141

42+
namespace testing {
43+
namespace internal {
44+
template <typename T>
45+
using identity_t = T;
46+
} // namespace internal
47+
} // namespace testing
48+
4249
#define MOCK_METHOD(...) \
4350
GMOCK_PP_VARIADIC_CALL(GMOCK_INTERNAL_MOCK_METHOD_ARG_, __VA_ARGS__)
4451

@@ -207,10 +214,24 @@
207214

208215
#define GMOCK_INTERNAL_IS_CALLTYPE_HELPER_Calltype
209216

210-
#define GMOCK_INTERNAL_SIGNATURE(_Ret, _Args) \
211-
GMOCK_PP_IF(GMOCK_PP_IS_BEGIN_PARENS(_Ret), GMOCK_PP_REMOVE_PARENS, \
212-
GMOCK_PP_IDENTITY) \
213-
(_Ret)(GMOCK_PP_FOR_EACH(GMOCK_INTERNAL_GET_TYPE, _, _Args))
217+
// Note: The use of `identity_t` here allows _Ret to represent return types that
218+
// would normally need to be specified in a different way. For example, a method
219+
// returning a function pointer must be written as
220+
//
221+
// fn_ptr_return_t (*method(method_args_t...))(fn_ptr_args_t...)
222+
//
223+
// But we only support placing the return type at the beginning. To handle this,
224+
// we wrap all calls in identity_t, so that a declaration will be expanded to
225+
//
226+
// identity_t<fn_ptr_return_t (*)(fn_ptr_args_t...)> method(method_args_t...)
227+
//
228+
// This allows us to work around the syntactic oddities of function/method
229+
// types.
230+
#define GMOCK_INTERNAL_SIGNATURE(_Ret, _Args) \
231+
::testing::internal::identity_t<GMOCK_PP_IF(GMOCK_PP_IS_BEGIN_PARENS(_Ret), \
232+
GMOCK_PP_REMOVE_PARENS, \
233+
GMOCK_PP_IDENTITY)(_Ret)>( \
234+
GMOCK_PP_FOR_EACH(GMOCK_INTERNAL_GET_TYPE, _, _Args))
214235

215236
#define GMOCK_INTERNAL_GET_TYPE(_i, _, _elem) \
216237
GMOCK_PP_COMMA_IF(_i) \

googlemock/test/gmock-function-mocker_test.cc

+7
Original file line numberDiff line numberDiff line change
@@ -101,6 +101,10 @@ class FooInterface {
101101
virtual int TypeWithComma(const std::map<int, std::string>& a_map) = 0;
102102
virtual int TypeWithTemplatedCopyCtor(const TemplatedCopyable<int>&) = 0;
103103

104+
virtual int (*ReturnsFunctionPointer1(int))(bool) = 0;
105+
using fn_ptr = int (*)(bool);
106+
virtual fn_ptr ReturnsFunctionPointer2(int) = 0;
107+
104108
#if GTEST_OS_WINDOWS
105109
STDMETHOD_(int, CTNullary)() = 0;
106110
STDMETHOD_(bool, CTUnary)(int x) = 0;
@@ -159,6 +163,9 @@ class MockFoo : public FooInterface {
159163
MOCK_METHOD(int, TypeWithTemplatedCopyCtor,
160164
(const TemplatedCopyable<int>&)); // NOLINT
161165

166+
MOCK_METHOD(int (*)(bool), ReturnsFunctionPointer1, (int), ());
167+
MOCK_METHOD(fn_ptr, ReturnsFunctionPointer2, (int), ());
168+
162169
#if GTEST_OS_WINDOWS
163170
MOCK_METHOD(int, CTNullary, (), (Calltype(STDMETHODCALLTYPE)));
164171
MOCK_METHOD(bool, CTUnary, (int), (Calltype(STDMETHODCALLTYPE)));

0 commit comments

Comments
 (0)