Index: third_party/grpc/src/ruby/ext/grpc/rb_channel_args.c |
diff --git a/third_party/grpc/src/ruby/ext/grpc/rb_channel_args.c b/third_party/grpc/src/ruby/ext/grpc/rb_channel_args.c |
new file mode 100644 |
index 0000000000000000000000000000000000000000..69827cea1c4263a2ca1211e3a67f11a573ee949a |
--- /dev/null |
+++ b/third_party/grpc/src/ruby/ext/grpc/rb_channel_args.c |
@@ -0,0 +1,169 @@ |
+/* |
+ * |
+ * Copyright 2015-2016, Google Inc. |
+ * All rights reserved. |
+ * |
+ * Redistribution and use in source and binary forms, with or without |
+ * modification, are permitted provided that the following conditions are |
+ * met: |
+ * |
+ * * Redistributions of source code must retain the above copyright |
+ * notice, this list of conditions and the following disclaimer. |
+ * * Redistributions in binary form must reproduce the above |
+ * copyright notice, this list of conditions and the following disclaimer |
+ * in the documentation and/or other materials provided with the |
+ * distribution. |
+ * * Neither the name of Google Inc. nor the names of its |
+ * contributors may be used to endorse or promote products derived from |
+ * this software without specific prior written permission. |
+ * |
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
+ * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
+ * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
+ * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
+ * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
+ * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
+ * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
+ * |
+ */ |
+ |
+#include <ruby/ruby.h> |
+#include "rb_grpc_imports.generated.h" |
+#include "rb_channel_args.h" |
+ |
+#include <ruby/ruby.h> |
+ |
+#include <grpc/grpc.h> |
+ |
+#include "rb_grpc.h" |
+ |
+static rb_data_type_t grpc_rb_channel_args_data_type = { |
+ "grpc_channel_args", |
+ {GRPC_RB_GC_NOT_MARKED, GRPC_RB_GC_DONT_FREE, GRPC_RB_MEMSIZE_UNAVAILABLE, |
+ {NULL, NULL}}, |
+ NULL, NULL, |
+#ifdef RUBY_TYPED_FREE_IMMEDIATELY |
+ RUBY_TYPED_FREE_IMMEDIATELY |
+#endif |
+}; |
+ |
+/* A callback the processes the hash key values in channel_args hash */ |
+static int grpc_rb_channel_create_in_process_add_args_hash_cb(VALUE key, |
+ VALUE val, |
+ VALUE args_obj) { |
+ const char* the_key; |
+ grpc_channel_args* args; |
+ |
+ switch (TYPE(key)) { |
+ case T_STRING: |
+ the_key = StringValuePtr(key); |
+ break; |
+ |
+ case T_SYMBOL: |
+ the_key = rb_id2name(SYM2ID(key)); |
+ break; |
+ |
+ default: |
+ rb_raise(rb_eTypeError, "bad chan arg: got <%s>, want <String|Symbol>", |
+ rb_obj_classname(key)); |
+ return ST_STOP; |
+ } |
+ |
+ TypedData_Get_Struct(args_obj, grpc_channel_args, |
+ &grpc_rb_channel_args_data_type, args); |
+ if (args->num_args <= 0) { |
+ rb_raise(rb_eRuntimeError, "hash_cb bug: num_args is %lu for key:%s", |
+ args->num_args, StringValueCStr(key)); |
+ return ST_STOP; |
+ } |
+ |
+ args->args[args->num_args - 1].key = (char*)the_key; |
+ switch (TYPE(val)) { |
+ case T_SYMBOL: |
+ args->args[args->num_args - 1].type = GRPC_ARG_STRING; |
+ args->args[args->num_args - 1].value.string = |
+ (char*)rb_id2name(SYM2ID(val)); |
+ --args->num_args; |
+ return ST_CONTINUE; |
+ |
+ case T_STRING: |
+ args->args[args->num_args - 1].type = GRPC_ARG_STRING; |
+ args->args[args->num_args - 1].value.string = StringValueCStr(val); |
+ --args->num_args; |
+ return ST_CONTINUE; |
+ |
+ case T_FIXNUM: |
+ args->args[args->num_args - 1].type = GRPC_ARG_INTEGER; |
+ args->args[args->num_args - 1].value.integer = NUM2INT(val); |
+ --args->num_args; |
+ return ST_CONTINUE; |
+ |
+ default: |
+ rb_raise(rb_eTypeError, "%s: bad value: got <%s>, want <String|Fixnum>", |
+ StringValueCStr(key), rb_obj_classname(val)); |
+ return ST_STOP; |
+ } |
+ rb_raise(rb_eRuntimeError, "impl bug: hash_cb reached to far while on key:%s", |
+ StringValueCStr(key)); |
+ return ST_STOP; |
+} |
+ |
+/* channel_convert_params allows the call to |
+ grpc_rb_hash_convert_to_channel_args to be made within an rb_protect |
+ exception-handler. This allows any allocated memory to be freed before |
+ propagating any exception that occurs */ |
+typedef struct channel_convert_params { |
+ VALUE src_hash; |
+ grpc_channel_args* dst; |
+} channel_convert_params; |
+ |
+static VALUE grpc_rb_hash_convert_to_channel_args0(VALUE as_value) { |
+ ID id_size = rb_intern("size"); |
+ VALUE grpc_rb_cChannelArgs = rb_define_class("TmpChannelArgs", rb_cObject); |
+ channel_convert_params* params = (channel_convert_params*)as_value; |
+ size_t num_args = 0; |
+ |
+ if (!NIL_P(params->src_hash) && TYPE(params->src_hash) != T_HASH) { |
+ rb_raise(rb_eTypeError, "bad channel args: got:<%s> want: a hash or nil", |
+ rb_obj_classname(params->src_hash)); |
+ return Qnil; |
+ } |
+ |
+ if (TYPE(params->src_hash) == T_HASH) { |
+ num_args = NUM2INT(rb_funcall(params->src_hash, id_size, 0)); |
+ params->dst->num_args = num_args; |
+ params->dst->args = ALLOC_N(grpc_arg, num_args); |
+ MEMZERO(params->dst->args, grpc_arg, num_args); |
+ rb_hash_foreach(params->src_hash, |
+ grpc_rb_channel_create_in_process_add_args_hash_cb, |
+ TypedData_Wrap_Struct(grpc_rb_cChannelArgs, |
+ &grpc_rb_channel_args_data_type, |
+ params->dst)); |
+ /* reset num_args as grpc_rb_channel_create_in_process_add_args_hash_cb |
+ * decrements it during has processing */ |
+ params->dst->num_args = num_args; |
+ } |
+ return Qnil; |
+} |
+ |
+void grpc_rb_hash_convert_to_channel_args(VALUE src_hash, |
+ grpc_channel_args* dst) { |
+ channel_convert_params params; |
+ int status = 0; |
+ |
+ /* Make a protected call to grpc_rb_hash_convert_channel_args */ |
+ params.src_hash = src_hash; |
+ params.dst = dst; |
+ rb_protect(grpc_rb_hash_convert_to_channel_args0, (VALUE) & params, &status); |
+ if (status != 0) { |
+ if (dst->args != NULL) { |
+ /* Free any allocated memory before propagating the error */ |
+ xfree(dst->args); |
+ } |
+ rb_jump_tag(status); |
+ } |
+} |