grpc 1.42.0.pre1-arm64-darwin → 1.58.3-arm64-darwin
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/grpc_c.64-ucrt.ruby +0 -0
- data/src/ruby/bin/math_pb.rb +24 -18
- data/src/ruby/ext/grpc/ext-export-truffleruby-with-ruby-abi-version.clang +2 -0
- data/src/ruby/ext/grpc/ext-export-truffleruby-with-ruby-abi-version.gcc +7 -0
- data/src/ruby/ext/grpc/ext-export-with-ruby-abi-version.clang +2 -0
- data/src/ruby/ext/grpc/ext-export-with-ruby-abi-version.gcc +7 -0
- data/src/ruby/ext/grpc/ext-export.gcc +1 -1
- data/src/ruby/ext/grpc/extconf.rb +117 -32
- data/src/ruby/ext/grpc/rb_call.c +63 -39
- data/src/ruby/ext/grpc/rb_call_credentials.c +0 -1
- data/src/ruby/ext/grpc/rb_channel.c +113 -84
- data/src/ruby/ext/grpc/rb_channel.h +1 -0
- data/src/ruby/ext/grpc/rb_channel_args.c +17 -2
- data/src/ruby/ext/grpc/rb_channel_args.h +4 -0
- data/src/ruby/ext/grpc/rb_channel_credentials.c +0 -1
- data/src/ruby/ext/grpc/rb_compression_options.c +1 -2
- data/src/ruby/ext/grpc/rb_event_thread.c +22 -6
- data/src/ruby/ext/grpc/rb_event_thread.h +1 -0
- data/src/ruby/ext/grpc/rb_grpc.c +193 -30
- data/src/ruby/ext/grpc/rb_grpc.h +8 -2
- data/src/ruby/ext/grpc/rb_grpc_imports.generated.c +66 -72
- data/src/ruby/ext/grpc/rb_grpc_imports.generated.h +102 -111
- data/src/ruby/ext/grpc/rb_loader.c +6 -2
- data/src/ruby/ext/grpc/rb_server.c +69 -49
- data/src/ruby/ext/grpc/rb_server_credentials.c +0 -1
- data/src/ruby/ext/grpc/rb_xds_channel_credentials.c +0 -1
- data/src/ruby/ext/grpc/rb_xds_server_credentials.c +0 -1
- data/src/ruby/lib/grpc/2.6/grpc_c.bundle +0 -0
- data/src/ruby/lib/grpc/2.7/grpc_c.bundle +0 -0
- data/src/ruby/lib/grpc/3.0/grpc_c.bundle +0 -0
- data/src/ruby/lib/grpc/3.1/grpc_c.bundle +0 -0
- data/src/ruby/lib/grpc/3.2/grpc_c.bundle +0 -0
- data/src/ruby/lib/grpc/errors.rb +1 -1
- data/src/ruby/lib/grpc/generic/active_call.rb +16 -15
- data/src/ruby/lib/grpc/generic/bidi_call.rb +4 -0
- data/src/ruby/lib/grpc/grpc.rb +1 -1
- data/src/ruby/lib/grpc/version.rb +1 -1
- data/src/ruby/pb/generate_proto_ruby.sh +1 -6
- data/src/ruby/pb/grpc/health/v1/health_pb.rb +24 -13
- data/src/ruby/pb/src/proto/grpc/testing/empty_pb.rb +24 -3
- data/src/ruby/pb/src/proto/grpc/testing/messages_pb.rb +26 -108
- data/src/ruby/pb/src/proto/grpc/testing/test_pb.rb +27 -3
- data/src/ruby/pb/test/client.rb +16 -0
- data/src/ruby/spec/channel_spec.rb +5 -43
- data/src/ruby/spec/client_server_spec.rb +20 -8
- data/src/ruby/spec/generic/active_call_spec.rb +12 -3
- data/src/ruby/spec/generic/client_stub_spec.rb +23 -23
- data/src/ruby/spec/generic/rpc_server_spec.rb +3 -3
- data/src/ruby/spec/generic/server_interceptors_spec.rb +1 -1
- data/src/ruby/spec/user_agent_spec.rb +1 -1
- metadata +61 -59
- data/src/ruby/lib/grpc/2.4/grpc_c.bundle +0 -0
- data/src/ruby/lib/grpc/2.5/grpc_c.bundle +0 -0
- /data/{grpc_c.32.ruby → grpc_c.32-msvcrt.ruby} +0 -0
- /data/{grpc_c.64.ruby → grpc_c.64-msvcrt.ruby} +0 -0
@@ -47,6 +47,9 @@ static ID id_channel;
|
|
47
47
|
* GCed before the channel */
|
48
48
|
static ID id_target;
|
49
49
|
|
50
|
+
/* hidden ivar that synchronizes post-fork channel re-creation */
|
51
|
+
static ID id_channel_recreation_mu;
|
52
|
+
|
50
53
|
/* id_insecure_channel is used to indicate that a channel is insecure */
|
51
54
|
static VALUE id_insecure_channel;
|
52
55
|
|
@@ -67,7 +70,7 @@ typedef struct bg_watched_channel {
|
|
67
70
|
/* grpc_rb_channel wraps a grpc_channel. */
|
68
71
|
typedef struct grpc_rb_channel {
|
69
72
|
VALUE credentials;
|
70
|
-
|
73
|
+
grpc_channel_args args;
|
71
74
|
/* The actual channel (protected in a wrapper to tell when it's safe to
|
72
75
|
* destroy) */
|
73
76
|
bg_watched_channel* bg_wrapped;
|
@@ -94,8 +97,6 @@ static bg_watched_channel* bg_watched_channel_list_head = NULL;
|
|
94
97
|
|
95
98
|
static void grpc_rb_channel_try_register_connection_polling(
|
96
99
|
bg_watched_channel* bg);
|
97
|
-
static void* wait_until_channel_polling_thread_started_no_gil(void*);
|
98
|
-
static void wait_until_channel_polling_thread_started_unblocking_func(void*);
|
99
100
|
static void* channel_init_try_register_connection_polling_without_gil(
|
100
101
|
void* arg);
|
101
102
|
|
@@ -104,11 +105,12 @@ typedef struct channel_init_try_register_stack {
|
|
104
105
|
grpc_rb_channel* wrapper;
|
105
106
|
} channel_init_try_register_stack;
|
106
107
|
|
107
|
-
static grpc_completion_queue*
|
108
|
+
static grpc_completion_queue* g_channel_polling_cq;
|
108
109
|
static gpr_mu global_connection_polling_mu;
|
109
110
|
static gpr_cv global_connection_polling_cv;
|
110
|
-
static int
|
111
|
-
static
|
111
|
+
static int g_abort_channel_polling = 0;
|
112
|
+
static gpr_once g_once_init = GPR_ONCE_INIT;
|
113
|
+
static VALUE g_channel_polling_thread = Qnil;
|
112
114
|
|
113
115
|
static int bg_watched_channel_list_lookup(bg_watched_channel* bg);
|
114
116
|
static bg_watched_channel* bg_watched_channel_list_create_and_add(
|
@@ -158,16 +160,13 @@ static void grpc_rb_channel_free_internal(void* p) {
|
|
158
160
|
* and we can count on this thread to not be interrupted or
|
159
161
|
* yield the gil. */
|
160
162
|
grpc_rb_channel_safe_destroy(ch->bg_wrapped);
|
161
|
-
ch->
|
163
|
+
grpc_rb_channel_args_destroy(&ch->args);
|
162
164
|
}
|
163
165
|
xfree(p);
|
164
166
|
}
|
165
167
|
|
166
168
|
/* Destroys Channel instances. */
|
167
|
-
static void grpc_rb_channel_free(void* p) {
|
168
|
-
grpc_rb_channel_free_internal(p);
|
169
|
-
grpc_ruby_shutdown();
|
170
|
-
}
|
169
|
+
static void grpc_rb_channel_free(void* p) { grpc_rb_channel_free_internal(p); }
|
171
170
|
|
172
171
|
/* Protects the mark object from GC */
|
173
172
|
static void grpc_rb_channel_mark(void* p) {
|
@@ -199,6 +198,7 @@ static VALUE grpc_rb_channel_alloc(VALUE cls) {
|
|
199
198
|
grpc_rb_channel* wrapper = ALLOC(grpc_rb_channel);
|
200
199
|
wrapper->bg_wrapped = NULL;
|
201
200
|
wrapper->credentials = Qnil;
|
201
|
+
MEMZERO(&wrapper->args, grpc_channel_args, 1);
|
202
202
|
return TypedData_Wrap_Struct(cls, &grpc_channel_data_type, wrapper);
|
203
203
|
}
|
204
204
|
|
@@ -218,31 +218,25 @@ static VALUE grpc_rb_channel_init(int argc, VALUE* argv, VALUE self) {
|
|
218
218
|
grpc_channel* ch = NULL;
|
219
219
|
grpc_channel_credentials* creds = NULL;
|
220
220
|
char* target_chars = NULL;
|
221
|
-
grpc_channel_args args;
|
222
221
|
channel_init_try_register_stack stack;
|
223
|
-
int stop_waiting_for_thread_start = 0;
|
224
|
-
MEMZERO(&args, grpc_channel_args, 1);
|
225
222
|
|
226
223
|
grpc_ruby_fork_guard();
|
227
|
-
rb_thread_call_without_gvl(
|
228
|
-
wait_until_channel_polling_thread_started_no_gil,
|
229
|
-
&stop_waiting_for_thread_start,
|
230
|
-
wait_until_channel_polling_thread_started_unblocking_func,
|
231
|
-
&stop_waiting_for_thread_start);
|
232
|
-
|
233
224
|
/* "3" == 3 mandatory args */
|
234
225
|
rb_scan_args(argc, argv, "3", &target, &channel_args, &credentials);
|
235
226
|
|
236
227
|
TypedData_Get_Struct(self, grpc_rb_channel, &grpc_channel_data_type, wrapper);
|
237
228
|
target_chars = StringValueCStr(target);
|
238
|
-
grpc_rb_hash_convert_to_channel_args(channel_args, &args);
|
229
|
+
grpc_rb_hash_convert_to_channel_args(channel_args, &wrapper->args);
|
239
230
|
if (TYPE(credentials) == T_SYMBOL) {
|
240
231
|
if (id_insecure_channel != SYM2ID(credentials)) {
|
241
232
|
rb_raise(rb_eTypeError,
|
242
233
|
"bad creds symbol, want :this_channel_is_insecure");
|
243
234
|
return Qnil;
|
244
235
|
}
|
245
|
-
|
236
|
+
grpc_channel_credentials* insecure_creds =
|
237
|
+
grpc_insecure_credentials_create();
|
238
|
+
ch = grpc_channel_create(target_chars, insecure_creds, &wrapper->args);
|
239
|
+
grpc_channel_credentials_release(insecure_creds);
|
246
240
|
} else {
|
247
241
|
wrapper->credentials = credentials;
|
248
242
|
if (grpc_rb_is_channel_credentials(credentials)) {
|
@@ -254,7 +248,7 @@ static VALUE grpc_rb_channel_init(int argc, VALUE* argv, VALUE self) {
|
|
254
248
|
"bad creds, want ChannelCredentials or XdsChannelCredentials");
|
255
249
|
return Qnil;
|
256
250
|
}
|
257
|
-
ch =
|
251
|
+
ch = grpc_channel_create(target_chars, creds, &wrapper->args);
|
258
252
|
}
|
259
253
|
|
260
254
|
GPR_ASSERT(ch);
|
@@ -263,16 +257,13 @@ static VALUE grpc_rb_channel_init(int argc, VALUE* argv, VALUE self) {
|
|
263
257
|
rb_thread_call_without_gvl(
|
264
258
|
channel_init_try_register_connection_polling_without_gil, &stack, NULL,
|
265
259
|
NULL);
|
266
|
-
|
267
|
-
if (args.args != NULL) {
|
268
|
-
xfree(args.args); /* Allocated by grpc_rb_hash_convert_to_channel_args */
|
269
|
-
}
|
270
260
|
if (ch == NULL) {
|
271
261
|
rb_raise(rb_eRuntimeError, "could not create an rpc channel to target:%s",
|
272
262
|
target_chars);
|
273
263
|
return Qnil;
|
274
264
|
}
|
275
265
|
rb_ivar_set(self, id_target, target);
|
266
|
+
rb_ivar_set(self, id_channel_recreation_mu, rb_mutex_new());
|
276
267
|
return self;
|
277
268
|
}
|
278
269
|
|
@@ -286,7 +277,6 @@ static void* get_state_without_gil(void* arg) {
|
|
286
277
|
get_state_stack* stack = (get_state_stack*)arg;
|
287
278
|
|
288
279
|
gpr_mu_lock(&global_connection_polling_mu);
|
289
|
-
GPR_ASSERT(abort_channel_polling || channel_polling_thread_started);
|
290
280
|
if (stack->bg->channel_destroyed) {
|
291
281
|
stack->out = GRPC_CHANNEL_SHUTDOWN;
|
292
282
|
} else {
|
@@ -343,7 +333,7 @@ static void* wait_for_watch_state_op_complete_without_gvl(void* arg) {
|
|
343
333
|
gpr_mu_lock(&global_connection_polling_mu);
|
344
334
|
// it's unsafe to do a "watch" after "channel polling abort" because the cq
|
345
335
|
// has been shut down.
|
346
|
-
if (
|
336
|
+
if (g_abort_channel_polling || stack->bg_wrapped->channel_destroyed) {
|
347
337
|
gpr_mu_unlock(&global_connection_polling_mu);
|
348
338
|
return (void*)0;
|
349
339
|
}
|
@@ -351,7 +341,7 @@ static void* wait_for_watch_state_op_complete_without_gvl(void* arg) {
|
|
351
341
|
op->op_type = WATCH_STATE_API;
|
352
342
|
grpc_channel_watch_connectivity_state(stack->bg_wrapped->channel,
|
353
343
|
stack->last_state, stack->deadline,
|
354
|
-
|
344
|
+
g_channel_polling_cq, op);
|
355
345
|
|
356
346
|
while (!op->op.api_callback_args.called_back) {
|
357
347
|
gpr_cv_wait(&global_connection_polling_cv, &global_connection_polling_mu,
|
@@ -415,6 +405,51 @@ static VALUE grpc_rb_channel_watch_connectivity_state(VALUE self,
|
|
415
405
|
return op_success ? Qtrue : Qfalse;
|
416
406
|
}
|
417
407
|
|
408
|
+
static void grpc_rb_channel_maybe_recreate_channel_after_fork(
|
409
|
+
grpc_rb_channel* wrapper, VALUE target) {
|
410
|
+
// TODO(apolcyn): maybe check if fork support is enabled here.
|
411
|
+
// The only way we can get bg->channel_destroyed without bg itself being
|
412
|
+
// NULL is if we destroyed the channel during GRPC::prefork.
|
413
|
+
bg_watched_channel* bg = wrapper->bg_wrapped;
|
414
|
+
if (bg->channel_destroyed) {
|
415
|
+
// There must be one ref at this point, held by the ruby-level channel
|
416
|
+
// object, drop this one here.
|
417
|
+
GPR_ASSERT(bg->refcount == 1);
|
418
|
+
rb_thread_call_without_gvl(channel_safe_destroy_without_gil, bg, NULL,
|
419
|
+
NULL);
|
420
|
+
// re-create C-core channel
|
421
|
+
const char* target_str = StringValueCStr(target);
|
422
|
+
grpc_channel* channel;
|
423
|
+
if (wrapper->credentials == Qnil) {
|
424
|
+
grpc_channel_credentials* insecure_creds =
|
425
|
+
grpc_insecure_credentials_create();
|
426
|
+
channel = grpc_channel_create(target_str, insecure_creds, &wrapper->args);
|
427
|
+
grpc_channel_credentials_release(insecure_creds);
|
428
|
+
} else {
|
429
|
+
grpc_channel_credentials* creds;
|
430
|
+
if (grpc_rb_is_channel_credentials(wrapper->credentials)) {
|
431
|
+
creds = grpc_rb_get_wrapped_channel_credentials(wrapper->credentials);
|
432
|
+
} else if (grpc_rb_is_xds_channel_credentials(wrapper->credentials)) {
|
433
|
+
creds =
|
434
|
+
grpc_rb_get_wrapped_xds_channel_credentials(wrapper->credentials);
|
435
|
+
} else {
|
436
|
+
rb_raise(rb_eTypeError,
|
437
|
+
"failed to re-create channel after fork: bad creds, want "
|
438
|
+
"ChannelCredentials or XdsChannelCredentials");
|
439
|
+
return;
|
440
|
+
}
|
441
|
+
channel = grpc_channel_create(target_str, creds, &wrapper->args);
|
442
|
+
}
|
443
|
+
// re-register with channel polling thread
|
444
|
+
channel_init_try_register_stack stack;
|
445
|
+
stack.channel = channel;
|
446
|
+
stack.wrapper = wrapper;
|
447
|
+
rb_thread_call_without_gvl(
|
448
|
+
channel_init_try_register_connection_polling_without_gil, &stack, NULL,
|
449
|
+
NULL);
|
450
|
+
}
|
451
|
+
}
|
452
|
+
|
418
453
|
/* Create a call given a grpc_channel, in order to call method. The request
|
419
454
|
is not sent until grpc_call_invoke is called. */
|
420
455
|
static VALUE grpc_rb_channel_create_call(VALUE self, VALUE parent, VALUE mask,
|
@@ -449,6 +484,11 @@ static VALUE grpc_rb_channel_create_call(VALUE self, VALUE parent, VALUE mask,
|
|
449
484
|
rb_raise(rb_eRuntimeError, "closed!");
|
450
485
|
return Qnil;
|
451
486
|
}
|
487
|
+
// TODO(apolcyn): only do this check if fork support is enabled
|
488
|
+
rb_mutex_lock(rb_ivar_get(self, id_channel_recreation_mu));
|
489
|
+
grpc_rb_channel_maybe_recreate_channel_after_fork(
|
490
|
+
wrapper, rb_ivar_get(self, id_target));
|
491
|
+
rb_mutex_unlock(rb_ivar_get(self, id_channel_recreation_mu));
|
452
492
|
|
453
493
|
cq = grpc_completion_queue_create_for_pluck(NULL);
|
454
494
|
method_slice =
|
@@ -578,19 +618,15 @@ static void grpc_rb_channel_try_register_connection_polling(
|
|
578
618
|
bg_watched_channel* bg) {
|
579
619
|
grpc_connectivity_state conn_state;
|
580
620
|
watch_state_op* op = NULL;
|
581
|
-
|
582
|
-
GPR_ASSERT(channel_polling_thread_started || abort_channel_polling);
|
583
|
-
|
584
621
|
if (bg->refcount == 0) {
|
585
622
|
GPR_ASSERT(bg->channel_destroyed);
|
586
623
|
bg_watched_channel_list_free_and_remove(bg);
|
587
624
|
return;
|
588
625
|
}
|
589
626
|
GPR_ASSERT(bg->refcount == 1);
|
590
|
-
if (bg->channel_destroyed ||
|
627
|
+
if (bg->channel_destroyed || g_abort_channel_polling) {
|
591
628
|
return;
|
592
629
|
}
|
593
|
-
|
594
630
|
conn_state = grpc_channel_check_connectivity_state(bg->channel, 0);
|
595
631
|
if (conn_state == GRPC_CHANNEL_SHUTDOWN) {
|
596
632
|
return;
|
@@ -598,13 +634,12 @@ static void grpc_rb_channel_try_register_connection_polling(
|
|
598
634
|
GPR_ASSERT(bg_watched_channel_list_lookup(bg));
|
599
635
|
// prevent bg from being free'd by GC while background thread is watching it
|
600
636
|
bg->refcount++;
|
601
|
-
|
602
637
|
op = gpr_zalloc(sizeof(watch_state_op));
|
603
638
|
op->op_type = CONTINUOUS_WATCH;
|
604
639
|
op->op.continuous_watch_callback_args.bg = bg;
|
605
640
|
grpc_channel_watch_connectivity_state(bg->channel, conn_state,
|
606
641
|
gpr_inf_future(GPR_CLOCK_REALTIME),
|
607
|
-
|
642
|
+
g_channel_polling_cq, op);
|
608
643
|
}
|
609
644
|
|
610
645
|
// Note this loop breaks out with a single call of
|
@@ -621,14 +656,12 @@ static void* run_poll_channels_loop_no_gil(void* arg) {
|
|
621
656
|
gpr_log(GPR_DEBUG, "GRPC_RUBY: run_poll_channels_loop_no_gil - begin");
|
622
657
|
|
623
658
|
gpr_mu_lock(&global_connection_polling_mu);
|
624
|
-
GPR_ASSERT(!channel_polling_thread_started);
|
625
|
-
channel_polling_thread_started = 1;
|
626
659
|
gpr_cv_broadcast(&global_connection_polling_cv);
|
627
660
|
gpr_mu_unlock(&global_connection_polling_mu);
|
628
661
|
|
629
662
|
for (;;) {
|
630
663
|
event = grpc_completion_queue_next(
|
631
|
-
|
664
|
+
g_channel_polling_cq, gpr_inf_future(GPR_CLOCK_REALTIME), NULL);
|
632
665
|
if (event.type == GRPC_QUEUE_SHUTDOWN) {
|
633
666
|
break;
|
634
667
|
}
|
@@ -649,7 +682,7 @@ static void* run_poll_channels_loop_no_gil(void* arg) {
|
|
649
682
|
}
|
650
683
|
gpr_mu_unlock(&global_connection_polling_mu);
|
651
684
|
}
|
652
|
-
grpc_completion_queue_destroy(
|
685
|
+
grpc_completion_queue_destroy(g_channel_polling_cq);
|
653
686
|
gpr_log(GPR_DEBUG,
|
654
687
|
"GRPC_RUBY: run_poll_channels_loop_no_gil - exit connection polling "
|
655
688
|
"loop");
|
@@ -666,11 +699,11 @@ static void run_poll_channels_loop_unblocking_func(void* arg) {
|
|
666
699
|
"GRPC_RUBY: run_poll_channels_loop_unblocking_func - begin aborting "
|
667
700
|
"connection polling");
|
668
701
|
// early out after first time through
|
669
|
-
if (
|
702
|
+
if (g_abort_channel_polling) {
|
670
703
|
gpr_mu_unlock(&global_connection_polling_mu);
|
671
704
|
return;
|
672
705
|
}
|
673
|
-
|
706
|
+
g_abort_channel_polling = 1;
|
674
707
|
|
675
708
|
// force pending watches to end by switching to shutdown state
|
676
709
|
bg = bg_watched_channel_list_head;
|
@@ -682,7 +715,9 @@ static void run_poll_channels_loop_unblocking_func(void* arg) {
|
|
682
715
|
bg = bg->next;
|
683
716
|
}
|
684
717
|
|
685
|
-
|
718
|
+
gpr_log(GPR_DEBUG, "GRPC_RUBY: cq shutdown on global polling cq. pid: %d",
|
719
|
+
getpid());
|
720
|
+
grpc_completion_queue_shutdown(g_channel_polling_cq);
|
686
721
|
gpr_cv_broadcast(&global_connection_polling_cv);
|
687
722
|
gpr_mu_unlock(&global_connection_polling_mu);
|
688
723
|
gpr_log(GPR_DEBUG,
|
@@ -696,47 +731,25 @@ static VALUE run_poll_channels_loop(VALUE arg) {
|
|
696
731
|
gpr_log(
|
697
732
|
GPR_DEBUG,
|
698
733
|
"GRPC_RUBY: run_poll_channels_loop - create connection polling thread");
|
699
|
-
grpc_ruby_init();
|
700
734
|
rb_thread_call_without_gvl(run_poll_channels_loop_no_gil, NULL,
|
701
735
|
run_poll_channels_loop_unblocking_func, NULL);
|
702
|
-
grpc_ruby_shutdown();
|
703
736
|
return Qnil;
|
704
737
|
}
|
705
738
|
|
706
|
-
static void* wait_until_channel_polling_thread_started_no_gil(void* arg) {
|
707
|
-
int* stop_waiting = (int*)arg;
|
708
|
-
gpr_log(GPR_DEBUG, "GRPC_RUBY: wait for channel polling thread to start");
|
709
|
-
gpr_mu_lock(&global_connection_polling_mu);
|
710
|
-
while (!channel_polling_thread_started && !abort_channel_polling &&
|
711
|
-
!*stop_waiting) {
|
712
|
-
gpr_cv_wait(&global_connection_polling_cv, &global_connection_polling_mu,
|
713
|
-
gpr_inf_future(GPR_CLOCK_REALTIME));
|
714
|
-
}
|
715
|
-
gpr_mu_unlock(&global_connection_polling_mu);
|
716
|
-
|
717
|
-
return NULL;
|
718
|
-
}
|
719
|
-
|
720
|
-
static void wait_until_channel_polling_thread_started_unblocking_func(
|
721
|
-
void* arg) {
|
722
|
-
int* stop_waiting = (int*)arg;
|
723
|
-
gpr_mu_lock(&global_connection_polling_mu);
|
724
|
-
gpr_log(GPR_DEBUG,
|
725
|
-
"GRPC_RUBY: interrupt wait for channel polling thread to start");
|
726
|
-
*stop_waiting = 1;
|
727
|
-
gpr_cv_broadcast(&global_connection_polling_cv);
|
728
|
-
gpr_mu_unlock(&global_connection_polling_mu);
|
729
|
-
}
|
730
|
-
|
731
739
|
static void* set_abort_channel_polling_without_gil(void* arg) {
|
732
740
|
(void)arg;
|
733
741
|
gpr_mu_lock(&global_connection_polling_mu);
|
734
|
-
|
742
|
+
g_abort_channel_polling = 1;
|
735
743
|
gpr_cv_broadcast(&global_connection_polling_cv);
|
736
744
|
gpr_mu_unlock(&global_connection_polling_mu);
|
737
745
|
return NULL;
|
738
746
|
}
|
739
747
|
|
748
|
+
static void do_basic_init() {
|
749
|
+
gpr_mu_init(&global_connection_polling_mu);
|
750
|
+
gpr_cv_init(&global_connection_polling_cv);
|
751
|
+
}
|
752
|
+
|
740
753
|
/* Temporary fix for
|
741
754
|
* https://github.com/GoogleCloudPlatform/google-cloud-ruby/issues/899.
|
742
755
|
* Transports in idle channels can get destroyed. Normally c-core re-connects,
|
@@ -748,23 +761,36 @@ static void* set_abort_channel_polling_without_gil(void* arg) {
|
|
748
761
|
* TODO(apolcyn) remove this when core handles new RPCs on dead connections.
|
749
762
|
*/
|
750
763
|
void grpc_rb_channel_polling_thread_start() {
|
751
|
-
|
752
|
-
|
753
|
-
GPR_ASSERT(!
|
754
|
-
GPR_ASSERT(
|
755
|
-
GPR_ASSERT(channel_polling_cq == NULL);
|
756
|
-
|
757
|
-
gpr_mu_init(&global_connection_polling_mu);
|
758
|
-
gpr_cv_init(&global_connection_polling_cv);
|
764
|
+
gpr_once_init(&g_once_init, do_basic_init);
|
765
|
+
GPR_ASSERT(!RTEST(g_channel_polling_thread));
|
766
|
+
GPR_ASSERT(!g_abort_channel_polling);
|
767
|
+
GPR_ASSERT(g_channel_polling_cq == NULL);
|
759
768
|
|
760
|
-
|
761
|
-
|
769
|
+
g_channel_polling_cq = grpc_completion_queue_create_for_next(NULL);
|
770
|
+
g_channel_polling_thread = rb_thread_create(run_poll_channels_loop, NULL);
|
762
771
|
|
763
|
-
if (!RTEST(
|
764
|
-
gpr_log(
|
772
|
+
if (!RTEST(g_channel_polling_thread)) {
|
773
|
+
gpr_log(GPR_ERROR, "GRPC_RUBY: failed to spawn channel polling thread");
|
765
774
|
rb_thread_call_without_gvl(set_abort_channel_polling_without_gil, NULL,
|
766
775
|
NULL, NULL);
|
776
|
+
return;
|
777
|
+
}
|
778
|
+
}
|
779
|
+
|
780
|
+
void grpc_rb_channel_polling_thread_stop() {
|
781
|
+
if (!RTEST(g_channel_polling_thread)) {
|
782
|
+
gpr_log(GPR_ERROR,
|
783
|
+
"GRPC_RUBY: channel polling thread stop: thread was not started");
|
784
|
+
return;
|
767
785
|
}
|
786
|
+
rb_thread_call_without_gvl(run_poll_channels_loop_unblocking_func, NULL, NULL,
|
787
|
+
NULL);
|
788
|
+
rb_funcall(g_channel_polling_thread, rb_intern("join"), 0);
|
789
|
+
// state associated with the channel polling thread is destroyed, reset so
|
790
|
+
// we can start again later
|
791
|
+
g_channel_polling_thread = Qnil;
|
792
|
+
g_abort_channel_polling = false;
|
793
|
+
g_channel_polling_cq = NULL;
|
768
794
|
}
|
769
795
|
|
770
796
|
static void Init_grpc_propagate_masks() {
|
@@ -800,7 +826,9 @@ static void Init_grpc_connectivity_states() {
|
|
800
826
|
}
|
801
827
|
|
802
828
|
void Init_grpc_channel() {
|
829
|
+
rb_global_variable(&g_channel_polling_thread);
|
803
830
|
grpc_rb_cChannelArgs = rb_define_class("TmpChannelArgs", rb_cObject);
|
831
|
+
rb_undef_alloc_func(grpc_rb_cChannelArgs);
|
804
832
|
grpc_rb_cChannel =
|
805
833
|
rb_define_class_under(grpc_rb_mGrpcCore, "Channel", rb_cObject);
|
806
834
|
|
@@ -825,6 +853,7 @@ void Init_grpc_channel() {
|
|
825
853
|
|
826
854
|
id_channel = rb_intern("__channel");
|
827
855
|
id_target = rb_intern("__target");
|
856
|
+
id_channel_recreation_mu = rb_intern("__channel_recreation_mu");
|
828
857
|
rb_define_const(grpc_rb_cChannel, "SSL_TARGET",
|
829
858
|
ID2SYM(rb_intern(GRPC_SSL_TARGET_NAME_OVERRIDE_ARG)));
|
830
859
|
rb_define_const(grpc_rb_cChannel, "ENABLE_CENSUS",
|
@@ -24,6 +24,7 @@
|
|
24
24
|
#include "rb_grpc_imports.generated.h"
|
25
25
|
|
26
26
|
#include <grpc/grpc.h>
|
27
|
+
#include <grpc/support/log.h>
|
27
28
|
|
28
29
|
static rb_data_type_t grpc_rb_channel_args_data_type = {
|
29
30
|
"grpc_channel_args",
|
@@ -73,13 +74,14 @@ static int grpc_rb_channel_create_in_process_add_args_hash_cb(VALUE key,
|
|
73
74
|
case T_SYMBOL:
|
74
75
|
args->args[args->num_args - 1].type = GRPC_ARG_STRING;
|
75
76
|
args->args[args->num_args - 1].value.string =
|
76
|
-
(
|
77
|
+
strdup(rb_id2name(SYM2ID(val)));
|
77
78
|
--args->num_args;
|
78
79
|
return ST_CONTINUE;
|
79
80
|
|
80
81
|
case T_STRING:
|
81
82
|
args->args[args->num_args - 1].type = GRPC_ARG_STRING;
|
82
|
-
args->args[args->num_args - 1].value.string =
|
83
|
+
args->args[args->num_args - 1].value.string =
|
84
|
+
strdup(StringValueCStr(val));
|
83
85
|
--args->num_args;
|
84
86
|
return ST_CONTINUE;
|
85
87
|
|
@@ -111,6 +113,7 @@ typedef struct channel_convert_params {
|
|
111
113
|
static VALUE grpc_rb_hash_convert_to_channel_args0(VALUE as_value) {
|
112
114
|
ID id_size = rb_intern("size");
|
113
115
|
VALUE grpc_rb_cChannelArgs = rb_define_class("TmpChannelArgs", rb_cObject);
|
116
|
+
rb_undef_alloc_func(grpc_rb_cChannelArgs);
|
114
117
|
channel_convert_params* params = (channel_convert_params*)as_value;
|
115
118
|
size_t num_args = 0;
|
116
119
|
|
@@ -153,3 +156,15 @@ void grpc_rb_hash_convert_to_channel_args(VALUE src_hash,
|
|
153
156
|
rb_jump_tag(status);
|
154
157
|
}
|
155
158
|
}
|
159
|
+
|
160
|
+
void grpc_rb_channel_args_destroy(grpc_channel_args* args) {
|
161
|
+
GPR_ASSERT(args != NULL);
|
162
|
+
if (args->args == NULL) return;
|
163
|
+
for (int i = 0; i < args->num_args; i++) {
|
164
|
+
if (args->args[i].type == GRPC_ARG_STRING) {
|
165
|
+
// we own string pointers, which were created with strdup
|
166
|
+
free(args->args[i].value.string);
|
167
|
+
}
|
168
|
+
}
|
169
|
+
xfree(args->args);
|
170
|
+
}
|
@@ -35,4 +35,8 @@
|
|
35
35
|
void grpc_rb_hash_convert_to_channel_args(VALUE src_hash,
|
36
36
|
grpc_channel_args* dst);
|
37
37
|
|
38
|
+
/* Destroys inner fields of args (does not deallocate the args pointer itself)
|
39
|
+
*/
|
40
|
+
void grpc_rb_channel_args_destroy(grpc_channel_args* args);
|
41
|
+
|
38
42
|
#endif /* GRPC_RB_CHANNEL_ARGS_H_ */
|
@@ -63,7 +63,6 @@ static void grpc_rb_channel_credentials_free_internal(void* p) {
|
|
63
63
|
/* Destroys the credentials instances. */
|
64
64
|
static void grpc_rb_channel_credentials_free(void* p) {
|
65
65
|
grpc_rb_channel_credentials_free_internal(p);
|
66
|
-
grpc_ruby_shutdown();
|
67
66
|
}
|
68
67
|
|
69
68
|
/* Protects the mark object from GC */
|
@@ -29,7 +29,7 @@
|
|
29
29
|
#include <grpc/compression.h>
|
30
30
|
#include <grpc/grpc.h>
|
31
31
|
#include <grpc/impl/codegen/compression_types.h>
|
32
|
-
#include <grpc/impl/
|
32
|
+
#include <grpc/impl/grpc_types.h>
|
33
33
|
#include <grpc/support/alloc.h>
|
34
34
|
#include <grpc/support/log.h>
|
35
35
|
#include <grpc/support/string_util.h>
|
@@ -70,7 +70,6 @@ static void grpc_rb_compression_options_free_internal(void* p) {
|
|
70
70
|
* wrapped grpc compression options. */
|
71
71
|
static void grpc_rb_compression_options_free(void* p) {
|
72
72
|
grpc_rb_compression_options_free_internal(p);
|
73
|
-
grpc_ruby_shutdown();
|
74
73
|
}
|
75
74
|
|
76
75
|
/* Ruby recognized data type for the CompressionOptions class. */
|
@@ -51,6 +51,8 @@ typedef struct grpc_rb_event_queue {
|
|
51
51
|
} grpc_rb_event_queue;
|
52
52
|
|
53
53
|
static grpc_rb_event_queue event_queue;
|
54
|
+
static VALUE g_event_thread = Qnil;
|
55
|
+
static bool g_one_time_init_done = false;
|
54
56
|
|
55
57
|
void grpc_rb_event_queue_enqueue(void (*callback)(void*), void* argument) {
|
56
58
|
grpc_rb_event* event = gpr_malloc(sizeof(grpc_rb_event));
|
@@ -117,7 +119,6 @@ static void grpc_rb_event_unblocking_func(void* arg) {
|
|
117
119
|
static VALUE grpc_rb_event_thread(VALUE arg) {
|
118
120
|
grpc_rb_event* event;
|
119
121
|
(void)arg;
|
120
|
-
grpc_ruby_init();
|
121
122
|
while (true) {
|
122
123
|
event = (grpc_rb_event*)rb_thread_call_without_gvl(
|
123
124
|
grpc_rb_wait_for_event_no_gil, NULL, grpc_rb_event_unblocking_func,
|
@@ -131,15 +132,30 @@ static VALUE grpc_rb_event_thread(VALUE arg) {
|
|
131
132
|
}
|
132
133
|
}
|
133
134
|
grpc_rb_event_queue_destroy();
|
134
|
-
grpc_ruby_shutdown();
|
135
135
|
return Qnil;
|
136
136
|
}
|
137
137
|
|
138
138
|
void grpc_rb_event_queue_thread_start() {
|
139
|
-
|
139
|
+
if (!g_one_time_init_done) {
|
140
|
+
g_one_time_init_done = true;
|
141
|
+
gpr_mu_init(&event_queue.mu);
|
142
|
+
gpr_cv_init(&event_queue.cv);
|
143
|
+
rb_global_variable(&g_event_thread);
|
144
|
+
event_queue.head = event_queue.tail = NULL;
|
145
|
+
}
|
140
146
|
event_queue.abort = false;
|
141
|
-
|
142
|
-
|
147
|
+
GPR_ASSERT(!RTEST(g_event_thread));
|
148
|
+
g_event_thread = rb_thread_create(grpc_rb_event_thread, NULL);
|
149
|
+
}
|
143
150
|
|
144
|
-
|
151
|
+
void grpc_rb_event_queue_thread_stop() {
|
152
|
+
GPR_ASSERT(g_one_time_init_done);
|
153
|
+
if (!RTEST(g_event_thread)) {
|
154
|
+
gpr_log(GPR_ERROR,
|
155
|
+
"GRPC_RUBY: call credentials thread stop: thread not running");
|
156
|
+
return;
|
157
|
+
}
|
158
|
+
rb_thread_call_without_gvl(grpc_rb_event_unblocking_func, NULL, NULL, NULL);
|
159
|
+
rb_funcall(g_event_thread, rb_intern("join"), 0);
|
160
|
+
g_event_thread = Qnil;
|
145
161
|
}
|