Skip to content

Commit 5741d75

Browse files
committed
Merge #34: Add shared_ptr callback support
9ce0335 Add comment saying how to fix clientInvoke missing Proxy.Context assert (Russell Yanofsky) 31b4f1b Add shared_ptr ownership and lifetime support (Russell Yanofsky) 27f8a35 Add saveCallback / callbackSaved test setup (Russell Yanofsky) 5390a1b Add support for passing shared_ptrs without extending lifetime (Russell Yanofsky) 39bbf74 Add CustomReadField priority param for more flexibility and consistency with CustomBuildField (Russell Yanofsky) da489be Fix bugs in PassField overload for callback objects passed by reference (Russell Yanofsky) ab4568b Add test coverage for thread map and callbacks (Russell Yanofsky) Pull request description: This is needed after bitcoin/bitcoin#18338 which changed `handleNotifications()` `Notifications&` callback argument to `std::shared_ptr<Notifications>`. Easiest way to support this was to change `ProxyServerBase` reference from a raw pointer to shared pointer and make necessary BuildField changes to pass along the `shared_ptr`. Alternative might have been to add more generic cleanup support to `ProxyServerBase` instead of hardcoding `shared_ptr`. The change also required making the ReadField callback overload more generic, which was a straightforward but kind of big change that touched a lot of code. There weren't any unit tests for callback support previously, so a lot of new test coverage was added. It includes coverage for `shared_ptr` lifetime correctness, making sure there's an IPC call decrementing server `shared_ptr` reference count when client `shared_ptr` proxy is reset. Top commit has no ACKs. Tree-SHA512: 96607bb339a5184ab34f9c133f7d5a9a6ac37a7a71187bdd6bd15235b44690cf19d8c09ad6e6966e51886cb00036cecf268406af7afd749b659970e9b00179ee
2 parents 1d630f5 + 9ce0335 commit 5741d75

File tree

6 files changed

+141
-38
lines changed

6 files changed

+141
-38
lines changed

include/mp/proxy-io.h

+11-15
Original file line numberDiff line numberDiff line change
@@ -412,28 +412,25 @@ ProxyClientBase<Interface, Impl>::~ProxyClientBase() noexcept
412412
}
413413

414414
template <typename Interface, typename Impl>
415-
ProxyServerBase<Interface, Impl>::ProxyServerBase(Impl* impl, bool owned, Connection& connection)
416-
: m_impl(impl), m_owned(owned), m_connection(connection)
415+
ProxyServerBase<Interface, Impl>::ProxyServerBase(std::shared_ptr<Impl> impl, Connection& connection)
416+
: m_impl(std::move(impl)), m_connection(connection)
417417
{
418-
assert(impl != nullptr);
418+
assert(m_impl);
419419
std::unique_lock<std::mutex> lock(m_connection.m_loop.m_mutex);
420420
m_connection.m_loop.addClient(lock);
421421
}
422422

423423
template <typename Interface, typename Impl>
424424
ProxyServerBase<Interface, Impl>::~ProxyServerBase()
425425
{
426-
if (Impl* impl = m_impl) {
426+
if (m_impl) {
427427
// If impl is non-null, it means client was not destroyed cleanly (was
428428
// killed or disconnected). Since client isn't providing thread to run
429429
// destructor on, run asynchronously. Do not run destructor on current
430430
// (event loop) thread since destructors could be making IPC calls or
431431
// doing expensive cleanup.
432-
if (m_owned) {
433-
m_connection.addAsyncCleanup([impl] { delete impl; });
434-
}
435-
m_impl = nullptr;
436-
m_owned = false;
432+
auto impl = std::move(m_impl);
433+
m_connection.addAsyncCleanup([impl]() mutable { impl.reset(); });
437434
}
438435
std::unique_lock<std::mutex> lock(m_connection.m_loop.m_mutex);
439436
m_connection.m_loop.removeClient(lock);
@@ -442,9 +439,7 @@ ProxyServerBase<Interface, Impl>::~ProxyServerBase()
442439
template <typename Interface, typename Impl>
443440
void ProxyServerBase<Interface, Impl>::invokeDestroy()
444441
{
445-
if (m_owned) delete m_impl;
446-
m_impl = nullptr;
447-
m_owned = false;
442+
m_impl.reset();
448443
}
449444

450445
struct ThreadContext
@@ -509,9 +504,10 @@ template <typename InitInterface, typename InitImpl>
509504
void _Serve(EventLoop& loop, kj::Own<kj::AsyncIoStream>&& stream, InitImpl& init)
510505
{
511506
loop.m_incoming_connections.emplace_front(loop, kj::mv(stream), [&](Connection& connection) {
512-
// Set owned to false so proxy object doesn't attempt to delete init
513-
// object on disconnect/close.
514-
return kj::heap<mp::ProxyServer<InitInterface>>(&init, false, connection);
507+
// Disable deleter so proxy server object doesn't attempt to delete the
508+
// init implementation when the proxy client is destroyed or
509+
// disconnected.
510+
return kj::heap<ProxyServer<InitInterface>>(std::shared_ptr<InitImpl>(&init, [](InitImpl*){}), connection);
515511
});
516512
auto it = loop.m_incoming_connections.begin();
517513
it->onDisconnect([&loop, it] {

include/mp/proxy-types.h

+70-13
Original file line numberDiff line numberDiff line change
@@ -235,6 +235,7 @@ struct ReadDestValue
235235

236236
template <typename LocalType, typename Input, typename ReadDest>
237237
decltype(auto) CustomReadField(TypeList<std::optional<LocalType>>,
238+
Priority<1>,
238239
InvokeContext& invoke_context,
239240
Input&& input,
240241
ReadDest&& read_dest)
@@ -256,6 +257,7 @@ decltype(auto) CustomReadField(TypeList<std::optional<LocalType>>,
256257

257258
template <typename LocalType, typename Input, typename ReadDest>
258259
decltype(auto) CustomReadField(TypeList<std::shared_ptr<LocalType>>,
260+
Priority<0>,
259261
InvokeContext& invoke_context,
260262
Input&& input,
261263
ReadDest&& read_dest)
@@ -277,6 +279,7 @@ decltype(auto) CustomReadField(TypeList<std::shared_ptr<LocalType>>,
277279

278280
template <typename LocalType, typename Input, typename ReadDest>
279281
decltype(auto) CustomReadField(TypeList<LocalType*>,
282+
Priority<1>,
280283
InvokeContext& invoke_context,
281284
Input&& input,
282285
ReadDest&& read_dest)
@@ -290,6 +293,7 @@ decltype(auto) CustomReadField(TypeList<LocalType*>,
290293

291294
template <typename LocalType, typename Input, typename ReadDest>
292295
decltype(auto) CustomReadField(TypeList<std::shared_ptr<const LocalType>>,
296+
Priority<1>,
293297
InvokeContext& invoke_context,
294298
Input&& input,
295299
ReadDest&& read_dest)
@@ -309,6 +313,7 @@ decltype(auto) CustomReadField(TypeList<std::shared_ptr<const LocalType>>,
309313

310314
template <typename LocalType, typename Input, typename ReadDest>
311315
decltype(auto) CustomReadField(TypeList<std::vector<LocalType>>,
316+
Priority<1>,
312317
InvokeContext& invoke_context,
313318
Input&& input,
314319
ReadDest&& read_dest)
@@ -329,6 +334,7 @@ decltype(auto) CustomReadField(TypeList<std::vector<LocalType>>,
329334

330335
template <typename LocalType, typename Input, typename ReadDest>
331336
decltype(auto) CustomReadField(TypeList<std::set<LocalType>>,
337+
Priority<1>,
332338
InvokeContext& invoke_context,
333339
Input&& input,
334340
ReadDest&& read_dest)
@@ -347,6 +353,7 @@ decltype(auto) CustomReadField(TypeList<std::set<LocalType>>,
347353

348354
template <typename KeyLocalType, typename ValueLocalType, typename Input, typename ReadDest>
349355
decltype(auto) CustomReadField(TypeList<std::map<KeyLocalType, ValueLocalType>>,
356+
Priority<1>,
350357
InvokeContext& invoke_context,
351358
Input&& input,
352359
ReadDest&& read_dest)
@@ -367,6 +374,7 @@ decltype(auto) CustomReadField(TypeList<std::map<KeyLocalType, ValueLocalType>>,
367374

368375
template <typename KeyLocalType, typename ValueLocalType, typename Input, typename ReadDest>
369376
decltype(auto) CustomReadField(TypeList<std::pair<KeyLocalType, ValueLocalType>>,
377+
Priority<1>,
370378
InvokeContext& invoke_context,
371379
Input&& input,
372380
ReadDest&& read_dest)
@@ -391,6 +399,7 @@ decltype(auto) CustomReadField(TypeList<std::pair<KeyLocalType, ValueLocalType>>
391399

392400
template <typename KeyLocalType, typename ValueLocalType, typename Input, typename ReadDest>
393401
decltype(auto) CustomReadField(TypeList<std::tuple<KeyLocalType, ValueLocalType>>,
402+
Priority<1>,
394403
InvokeContext& invoke_context,
395404
Input&& input,
396405
ReadDest&& read_dest)
@@ -407,6 +416,7 @@ decltype(auto) CustomReadField(TypeList<std::tuple<KeyLocalType, ValueLocalType>
407416

408417
template <typename LocalType, typename Input, typename ReadDest>
409418
decltype(auto) CustomReadField(TypeList<LocalType>,
419+
Priority<1>,
410420
InvokeContext& invoke_context,
411421
Input&& input,
412422
ReadDest&& read_dest,
@@ -417,6 +427,7 @@ decltype(auto) CustomReadField(TypeList<LocalType>,
417427

418428
template <typename LocalType, typename Input, typename ReadDest>
419429
decltype(auto) CustomReadField(TypeList<LocalType>,
430+
Priority<1>,
420431
InvokeContext& invoke_context,
421432
Input&& input,
422433
ReadDest&& read_dest,
@@ -431,6 +442,7 @@ decltype(auto) CustomReadField(TypeList<LocalType>,
431442

432443
template <typename LocalType, typename Input, typename ReadDest>
433444
decltype(auto) CustomReadField(TypeList<LocalType>,
445+
Priority<1>,
434446
InvokeContext& invoke_context,
435447
Input&& input,
436448
ReadDest&& read_dest,
@@ -443,6 +455,7 @@ decltype(auto) CustomReadField(TypeList<LocalType>,
443455

444456
template <typename Input, typename ReadDest>
445457
decltype(auto) CustomReadField(TypeList<std::string>,
458+
Priority<1>,
446459
InvokeContext& invoke_context,
447460
Input&& input,
448461
ReadDest&& read_dest)
@@ -453,6 +466,7 @@ decltype(auto) CustomReadField(TypeList<std::string>,
453466

454467
template <size_t size, typename Input, typename ReadDest>
455468
decltype(auto) CustomReadField(TypeList<unsigned char[size]>,
469+
Priority<1>,
456470
InvokeContext& invoke_context,
457471
Input&& input,
458472
ReadDest&& read_dest)
@@ -478,6 +492,23 @@ std::unique_ptr<Impl> CustomMakeProxyClient(InvokeContext& context, typename Int
478492

479493
template <typename LocalType, typename Input, typename ReadDest>
480494
decltype(auto) CustomReadField(TypeList<std::unique_ptr<LocalType>>,
495+
Priority<1>,
496+
InvokeContext& invoke_context,
497+
Input&& input,
498+
ReadDest&& read_dest,
499+
typename Decay<decltype(input.get())>::Calls* enable = nullptr)
500+
{
501+
using Interface = typename Decay<decltype(input.get())>::Calls;
502+
if (input.has()) {
503+
return read_dest.construct(
504+
CustomMakeProxyClient<Interface, LocalType>(invoke_context, std::move(input.get())));
505+
}
506+
return read_dest.construct();
507+
}
508+
509+
template <typename LocalType, typename Input, typename ReadDest>
510+
decltype(auto) CustomReadField(TypeList<std::shared_ptr<LocalType>>,
511+
Priority<1>,
481512
InvokeContext& invoke_context,
482513
Input&& input,
483514
ReadDest&& read_dest,
@@ -505,6 +536,7 @@ struct ProxyCallFn
505536

506537
template <typename FnR, typename... FnParams, typename Input, typename ReadDest>
507538
decltype(auto) CustomReadField(TypeList<std::function<FnR(FnParams...)>>,
539+
Priority<1>,
508540
InvokeContext& invoke_context,
509541
Input&& input,
510542
ReadDest&& read_dest)
@@ -546,6 +578,7 @@ void ReadOne(TypeList<LocalType> param,
546578

547579
template <typename LocalType, typename Input, typename ReadDest>
548580
decltype(auto) CustomReadField(TypeList<LocalType> param,
581+
Priority<1>,
549582
InvokeContext& invoke_context,
550583
Input&& input,
551584
ReadDest&& read_dest,
@@ -557,7 +590,7 @@ decltype(auto) CustomReadField(TypeList<LocalType> param,
557590
template <typename... LocalTypes, typename... Args>
558591
void ReadField(TypeList<LocalTypes...>, Args&&... args)
559592
{
560-
CustomReadField(TypeList<RemoveCvRef<LocalTypes>...>(), std::forward<Args>(args)...);
593+
CustomReadField(TypeList<RemoveCvRef<LocalTypes>...>(), Priority<2>(), std::forward<Args>(args)...);
561594
}
562595

563596
template <typename LocalType, typename Input>
@@ -642,18 +675,18 @@ void CustomBuildField(TypeList<std::function<FnR(FnParams...)>>,
642675
using Interface = typename decltype(output.get())::Calls;
643676
using Callback = ProxyCallbackImpl<FnR, FnParams...>;
644677
output.set(kj::heap<ProxyServer<Interface>>(
645-
new Callback(std::forward<Value>(value)), true /* owned */, invoke_context.connection));
678+
std::make_shared<Callback>(std::forward<Value>(value)), invoke_context.connection));
646679
}
647680
}
648681

649682
template <typename Interface, typename Impl>
650-
kj::Own<typename Interface::Server> MakeProxyServer(InvokeContext& context, std::unique_ptr<Impl>&& impl)
683+
kj::Own<typename Interface::Server> MakeProxyServer(InvokeContext& context, std::shared_ptr<Impl> impl)
651684
{
652-
return kj::heap<ProxyServer<Interface>>(impl.release(), true /* owned */, context.connection);
685+
return kj::heap<ProxyServer<Interface>>(std::move(impl), context.connection);
653686
}
654687

655688
template <typename Interface, typename Impl>
656-
kj::Own<typename Interface::Server> CustomMakeProxyServer(InvokeContext& context, std::unique_ptr<Impl>&& impl)
689+
kj::Own<typename Interface::Server> CustomMakeProxyServer(InvokeContext& context, std::shared_ptr<Impl>&& impl)
657690
{
658691
return MakeProxyServer<Interface, Impl>(context, std::move(impl));
659692
}
@@ -665,25 +698,40 @@ void CustomBuildField(TypeList<std::unique_ptr<Impl>>,
665698
Value&& value,
666699
Output&& output,
667700
typename Decay<decltype(output.get())>::Calls* enable = nullptr)
701+
{
702+
if (value) {
703+
using Interface = typename decltype(output.get())::Calls;
704+
output.set(CustomMakeProxyServer<Interface, Impl>(invoke_context, std::shared_ptr<Impl>(value.release())));
705+
}
706+
}
707+
708+
template <typename Impl, typename Value, typename Output>
709+
void CustomBuildField(TypeList<std::shared_ptr<Impl>>,
710+
Priority<2>,
711+
InvokeContext& invoke_context,
712+
Value&& value,
713+
Output&& output,
714+
typename Decay<decltype(output.get())>::Calls* enable = nullptr)
668715
{
669716
if (value) {
670717
using Interface = typename decltype(output.get())::Calls;
671718
output.set(CustomMakeProxyServer<Interface, Impl>(invoke_context, std::move(value)));
672719
}
673720
}
674721

675-
template <typename LocalType, typename Output>
676-
void CustomBuildField(TypeList<LocalType&>,
722+
template <typename Impl, typename Output>
723+
void CustomBuildField(TypeList<Impl&>,
677724
Priority<1>,
678725
InvokeContext& invoke_context,
679-
LocalType& value,
726+
Impl& value,
680727
Output&& output,
681728
typename decltype(output.get())::Calls* enable = nullptr)
682729
{
683-
// Set owned to false so proxy object doesn't attempt to delete interface
684-
// reference when it is discarded remotely, or on disconnect.
685-
output.set(kj::heap<ProxyServer<typename decltype(output.get())::Calls>>(
686-
&value, false /* owned */, invoke_context.connection));
730+
// Disable deleter so proxy server object doesn't attempt to delete the
731+
// wrapped implementation when the proxy client is destroyed or
732+
// disconnected.
733+
using Interface = typename decltype(output.get())::Calls;
734+
output.set(CustomMakeProxyServer<Interface, Impl>(invoke_context, std::shared_ptr<Impl>(&value, [](Impl*){})));
687735
}
688736

689737
template <typename LocalType, typename Value, typename Output>
@@ -977,7 +1025,7 @@ auto PassField(TypeList<LocalType&>, ServerContext& server_context, Fn&& fn, Arg
9771025
const auto& params = server_context.call_context.getParams();
9781026
const auto& input = Make<StructField, Accessor>(params);
9791027
using Interface = typename Decay<decltype(input.get())>::Calls;
980-
auto param = std::make_unique<ProxyClient<Interface>>(input.get(), *server_context.proxy_server.m_connection);
1028+
auto param = std::make_unique<ProxyClient<Interface>>(input.get(), &server_context.proxy_server.m_connection, false);
9811029
fn.invoke(server_context, std::forward<Args>(args)..., *param);
9821030
}
9831031

@@ -1053,6 +1101,7 @@ void CustomBuildField(TypeList<>,
10531101

10541102
template <typename Input>
10551103
decltype(auto) CustomReadField(TypeList<>,
1104+
Priority<1>,
10561105
InvokeContext& invoke_context,
10571106
Input&& input,
10581107
typename std::enable_if<std::is_same<decltype(input.get()), ThreadMap::Client>::value>::type* enable = nullptr)
@@ -1325,6 +1374,14 @@ void clientInvoke(ProxyClient& proxy_client, const GetRequest& get_request, Fiel
13251374
if (!g_thread_context.waiter) {
13261375
assert(g_thread_context.thread_name.empty());
13271376
g_thread_context.thread_name = ThreadName(proxy_client.m_connection->m_loop.m_exe_name);
1377+
// If next assert triggers, it means clientInvoke is being called from
1378+
// the capnp event loop thread. This can happen when a ProxyServer
1379+
// method implementation that runs synchronously on the event loop
1380+
// thread tries to make a blocking callback to the client. Any server
1381+
// method that makes a blocking callback or blocks in general needs to
1382+
// run asynchronously off the event loop thread. This is easy to fix by
1383+
// just adding a 'context :Proxy.Context' argument to the capnp method
1384+
// declaration so the server method runs in a dedicated thread.
13281385
assert(!g_thread_context.loop_thread);
13291386
g_thread_context.waiter = std::make_unique<Waiter>();
13301387
proxy_client.m_connection->m_loop.logPlain()

include/mp/proxy.h

+7-8
Original file line numberDiff line numberDiff line change
@@ -82,16 +82,15 @@ struct ProxyServerBase : public virtual Interface_::Server
8282
using Interface = Interface_;
8383
using Impl = Impl_;
8484

85-
ProxyServerBase(Impl* impl, bool owned, Connection& connection);
85+
ProxyServerBase(std::shared_ptr<Impl> impl, Connection& connection);
8686
virtual ~ProxyServerBase();
8787
void invokeDestroy();
8888

89-
Impl* m_impl;
9089
/**
91-
* Whether or not to delete native interface pointer when this capnp server
92-
* goes out of scope. This is true for servers created to wrap
93-
* unique_ptr<Impl> method arguments, but false for servers created to wrap
94-
* Impl& method arguments.
90+
* Implementation pointer that may or may not be owned and deleted when this
91+
* capnp server goes out of scope. It is owned for servers created to wrap
92+
* unique_ptr<Impl> method arguments, but unowned for servers created to
93+
* wrap Impl& method arguments.
9594
*
9695
* In the case of Impl& arguments, custom code is required on other side of
9796
* the connection to delete the capnp client & server objects since native
@@ -100,7 +99,7 @@ struct ProxyServerBase : public virtual Interface_::Server
10099
* this is implemented with addCloseHook callbacks to delete clients at
101100
* appropriate times depending on semantics of the particular method being
102101
* wrapped. */
103-
bool m_owned;
102+
std::shared_ptr<Impl> m_impl;
104103
Connection& m_connection;
105104
};
106105

@@ -159,7 +158,7 @@ struct ProxyMethodTraits<MethodParams, Require<decltype(ProxyMethod<MethodParams
159158
{
160159
template <typename ServerContext, typename... Args>
161160
static auto invoke(ServerContext& server_context, Args&&... args) -> AUTO_RETURN(
162-
(server_context.proxy_server.m_impl->*ProxyMethod<MethodParams>::impl)(std::forward<Args>(args)...))
161+
(server_context.proxy_server.m_impl.get()->*ProxyMethod<MethodParams>::impl)(std::forward<Args>(args)...))
163162
};
164163

165164
//! Customizable (through template specialization) traits class used in generated ProxyClient implementations from

src/mp/test/foo.capnp

+11
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,17 @@ interface FooInterface $Proxy.wrap("mp::test::FooImplementation") {
1414
mapSize @1 (map :List(Pair(Text, Text))) -> (result :Int32);
1515
pass @2 (arg :FooStruct) -> (result :FooStruct);
1616
raise @3 (arg :FooStruct) -> (error :FooStruct $Proxy.exception("mp::test::FooStruct"));
17+
initThreadMap @4 (threadMap: Proxy.ThreadMap) -> (threadMap :Proxy.ThreadMap);
18+
callback @5 (context :Proxy.Context, callback :FooCallback, arg: Int32) -> (result :Int32);
19+
callbackUnique @6 (context :Proxy.Context, callback :FooCallback, arg: Int32) -> (result :Int32);
20+
callbackShared @7 (context :Proxy.Context, callback :FooCallback, arg: Int32) -> (result :Int32);
21+
saveCallback @8 (context :Proxy.Context, callback :FooCallback) -> ();
22+
callbackSaved @9 (context :Proxy.Context, arg: Int32) -> (result :Int32);
23+
}
24+
25+
interface FooCallback $Proxy.wrap("mp::test::FooCallback") {
26+
destroy @0 (context :Proxy.Context) -> ();
27+
call @1 (context :Proxy.Context, arg :Int32) -> (result :Int32);
1728
}
1829

1930
struct FooStruct $Proxy.wrap("mp::test::FooStruct") {

0 commit comments

Comments
 (0)