@@ -235,6 +235,7 @@ struct ReadDestValue
235
235
236
236
template <typename LocalType, typename Input, typename ReadDest>
237
237
decltype (auto ) CustomReadField(TypeList<std::optional<LocalType>>,
238
+ Priority<1 >,
238
239
InvokeContext& invoke_context,
239
240
Input&& input,
240
241
ReadDest&& read_dest)
@@ -256,6 +257,7 @@ decltype(auto) CustomReadField(TypeList<std::optional<LocalType>>,
256
257
257
258
template <typename LocalType, typename Input, typename ReadDest>
258
259
decltype (auto ) CustomReadField(TypeList<std::shared_ptr<LocalType>>,
260
+ Priority<0 >,
259
261
InvokeContext& invoke_context,
260
262
Input&& input,
261
263
ReadDest&& read_dest)
@@ -277,6 +279,7 @@ decltype(auto) CustomReadField(TypeList<std::shared_ptr<LocalType>>,
277
279
278
280
template <typename LocalType, typename Input, typename ReadDest>
279
281
decltype (auto ) CustomReadField(TypeList<LocalType*>,
282
+ Priority<1 >,
280
283
InvokeContext& invoke_context,
281
284
Input&& input,
282
285
ReadDest&& read_dest)
@@ -290,6 +293,7 @@ decltype(auto) CustomReadField(TypeList<LocalType*>,
290
293
291
294
template <typename LocalType, typename Input, typename ReadDest>
292
295
decltype (auto ) CustomReadField(TypeList<std::shared_ptr<const LocalType>>,
296
+ Priority<1 >,
293
297
InvokeContext& invoke_context,
294
298
Input&& input,
295
299
ReadDest&& read_dest)
@@ -309,6 +313,7 @@ decltype(auto) CustomReadField(TypeList<std::shared_ptr<const LocalType>>,
309
313
310
314
template <typename LocalType, typename Input, typename ReadDest>
311
315
decltype (auto ) CustomReadField(TypeList<std::vector<LocalType>>,
316
+ Priority<1 >,
312
317
InvokeContext& invoke_context,
313
318
Input&& input,
314
319
ReadDest&& read_dest)
@@ -329,6 +334,7 @@ decltype(auto) CustomReadField(TypeList<std::vector<LocalType>>,
329
334
330
335
template <typename LocalType, typename Input, typename ReadDest>
331
336
decltype (auto ) CustomReadField(TypeList<std::set<LocalType>>,
337
+ Priority<1 >,
332
338
InvokeContext& invoke_context,
333
339
Input&& input,
334
340
ReadDest&& read_dest)
@@ -347,6 +353,7 @@ decltype(auto) CustomReadField(TypeList<std::set<LocalType>>,
347
353
348
354
template <typename KeyLocalType, typename ValueLocalType, typename Input, typename ReadDest>
349
355
decltype (auto ) CustomReadField(TypeList<std::map<KeyLocalType, ValueLocalType>>,
356
+ Priority<1 >,
350
357
InvokeContext& invoke_context,
351
358
Input&& input,
352
359
ReadDest&& read_dest)
@@ -367,6 +374,7 @@ decltype(auto) CustomReadField(TypeList<std::map<KeyLocalType, ValueLocalType>>,
367
374
368
375
template <typename KeyLocalType, typename ValueLocalType, typename Input, typename ReadDest>
369
376
decltype (auto ) CustomReadField(TypeList<std::pair<KeyLocalType, ValueLocalType>>,
377
+ Priority<1 >,
370
378
InvokeContext& invoke_context,
371
379
Input&& input,
372
380
ReadDest&& read_dest)
@@ -391,6 +399,7 @@ decltype(auto) CustomReadField(TypeList<std::pair<KeyLocalType, ValueLocalType>>
391
399
392
400
template <typename KeyLocalType, typename ValueLocalType, typename Input, typename ReadDest>
393
401
decltype (auto ) CustomReadField(TypeList<std::tuple<KeyLocalType, ValueLocalType>>,
402
+ Priority<1 >,
394
403
InvokeContext& invoke_context,
395
404
Input&& input,
396
405
ReadDest&& read_dest)
@@ -407,6 +416,7 @@ decltype(auto) CustomReadField(TypeList<std::tuple<KeyLocalType, ValueLocalType>
407
416
408
417
template <typename LocalType, typename Input, typename ReadDest>
409
418
decltype (auto ) CustomReadField(TypeList<LocalType>,
419
+ Priority<1 >,
410
420
InvokeContext& invoke_context,
411
421
Input&& input,
412
422
ReadDest&& read_dest,
@@ -417,6 +427,7 @@ decltype(auto) CustomReadField(TypeList<LocalType>,
417
427
418
428
template <typename LocalType, typename Input, typename ReadDest>
419
429
decltype (auto ) CustomReadField(TypeList<LocalType>,
430
+ Priority<1 >,
420
431
InvokeContext& invoke_context,
421
432
Input&& input,
422
433
ReadDest&& read_dest,
@@ -431,6 +442,7 @@ decltype(auto) CustomReadField(TypeList<LocalType>,
431
442
432
443
template <typename LocalType, typename Input, typename ReadDest>
433
444
decltype (auto ) CustomReadField(TypeList<LocalType>,
445
+ Priority<1 >,
434
446
InvokeContext& invoke_context,
435
447
Input&& input,
436
448
ReadDest&& read_dest,
@@ -443,6 +455,7 @@ decltype(auto) CustomReadField(TypeList<LocalType>,
443
455
444
456
template <typename Input, typename ReadDest>
445
457
decltype (auto ) CustomReadField(TypeList<std::string>,
458
+ Priority<1 >,
446
459
InvokeContext& invoke_context,
447
460
Input&& input,
448
461
ReadDest&& read_dest)
@@ -453,6 +466,7 @@ decltype(auto) CustomReadField(TypeList<std::string>,
453
466
454
467
template <size_t size, typename Input, typename ReadDest>
455
468
decltype (auto ) CustomReadField(TypeList<unsigned char [size]>,
469
+ Priority<1 >,
456
470
InvokeContext& invoke_context,
457
471
Input&& input,
458
472
ReadDest&& read_dest)
@@ -478,6 +492,23 @@ std::unique_ptr<Impl> CustomMakeProxyClient(InvokeContext& context, typename Int
478
492
479
493
template <typename LocalType, typename Input, typename ReadDest>
480
494
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 >,
481
512
InvokeContext& invoke_context,
482
513
Input&& input,
483
514
ReadDest&& read_dest,
@@ -505,6 +536,7 @@ struct ProxyCallFn
505
536
506
537
template <typename FnR, typename ... FnParams, typename Input, typename ReadDest>
507
538
decltype (auto ) CustomReadField(TypeList<std::function<FnR(FnParams...)>>,
539
+ Priority<1>,
508
540
InvokeContext& invoke_context,
509
541
Input&& input,
510
542
ReadDest&& read_dest)
@@ -546,6 +578,7 @@ void ReadOne(TypeList<LocalType> param,
546
578
547
579
template <typename LocalType, typename Input, typename ReadDest>
548
580
decltype (auto ) CustomReadField(TypeList<LocalType> param,
581
+ Priority<1 >,
549
582
InvokeContext& invoke_context,
550
583
Input&& input,
551
584
ReadDest&& read_dest,
@@ -557,7 +590,7 @@ decltype(auto) CustomReadField(TypeList<LocalType> param,
557
590
template <typename ... LocalTypes, typename ... Args>
558
591
void ReadField (TypeList<LocalTypes...>, Args&&... args)
559
592
{
560
- CustomReadField (TypeList<RemoveCvRef<LocalTypes>...>(), std::forward<Args>(args)...);
593
+ CustomReadField (TypeList<RemoveCvRef<LocalTypes>...>(), Priority< 2 >(), std::forward<Args>(args)...);
561
594
}
562
595
563
596
template <typename LocalType, typename Input>
@@ -642,18 +675,18 @@ void CustomBuildField(TypeList<std::function<FnR(FnParams...)>>,
642
675
using Interface = typename decltype (output.get ())::Calls;
643
676
using Callback = ProxyCallbackImpl<FnR, FnParams...>;
644
677
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 ));
646
679
}
647
680
}
648
681
649
682
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)
651
684
{
652
- return kj::heap<ProxyServer<Interface>>(impl. release (), true /* owned */ , context.connection );
685
+ return kj::heap<ProxyServer<Interface>>(std::move (impl) , context.connection );
653
686
}
654
687
655
688
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)
657
690
{
658
691
return MakeProxyServer<Interface, Impl>(context, std::move (impl));
659
692
}
@@ -665,25 +698,40 @@ void CustomBuildField(TypeList<std::unique_ptr<Impl>>,
665
698
Value&& value,
666
699
Output&& output,
667
700
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)
668
715
{
669
716
if (value) {
670
717
using Interface = typename decltype (output.get ())::Calls;
671
718
output.set (CustomMakeProxyServer<Interface, Impl>(invoke_context, std::move (value)));
672
719
}
673
720
}
674
721
675
- template <typename LocalType , typename Output>
676
- void CustomBuildField (TypeList<LocalType &>,
722
+ template <typename Impl , typename Output>
723
+ void CustomBuildField (TypeList<Impl &>,
677
724
Priority<1 >,
678
725
InvokeContext& invoke_context,
679
- LocalType & value,
726
+ Impl & value,
680
727
Output&& output,
681
728
typename decltype (output.get())::Calls* enable = nullptr)
682
729
{
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*){})));
687
735
}
688
736
689
737
template <typename LocalType, typename Value, typename Output>
@@ -977,7 +1025,7 @@ auto PassField(TypeList<LocalType&>, ServerContext& server_context, Fn&& fn, Arg
977
1025
const auto & params = server_context.call_context .getParams ();
978
1026
const auto & input = Make<StructField, Accessor>(params);
979
1027
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 );
981
1029
fn.invoke (server_context, std::forward<Args>(args)..., *param);
982
1030
}
983
1031
@@ -1053,6 +1101,7 @@ void CustomBuildField(TypeList<>,
1053
1101
1054
1102
template <typename Input>
1055
1103
decltype (auto ) CustomReadField(TypeList<>,
1104
+ Priority<1 >,
1056
1105
InvokeContext& invoke_context,
1057
1106
Input&& input,
1058
1107
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
1325
1374
if (!g_thread_context.waiter ) {
1326
1375
assert (g_thread_context.thread_name .empty ());
1327
1376
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.
1328
1385
assert (!g_thread_context.loop_thread );
1329
1386
g_thread_context.waiter = std::make_unique<Waiter>();
1330
1387
proxy_client.m_connection ->m_loop .logPlain ()
0 commit comments