/* * * Copyright 2015, 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 "rb_channel_args.h" #include #include #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 ", 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 ", 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); } }