/* * MessagePack for Ruby * * Copyright (C) 2008-2015 Sadayuki Furuhashi * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifndef MSGPACK_RUBY_PACKER_EXT_REGISTRY_H__ #define MSGPACK_RUBY_PACKER_EXT_REGISTRY_H__ #include "compat.h" #include "ruby.h" struct msgpack_packer_ext_registry_t; typedef struct msgpack_packer_ext_registry_t msgpack_packer_ext_registry_t; struct msgpack_packer_ext_registry_t { VALUE hash; VALUE cache; // lookup cache for ext types inherited from a super class }; void msgpack_packer_ext_registry_static_init(); void msgpack_packer_ext_registry_static_destroy(); void msgpack_packer_ext_registry_init(msgpack_packer_ext_registry_t* pkrg); static inline void msgpack_packer_ext_registry_destroy(msgpack_packer_ext_registry_t* pkrg) { } void msgpack_packer_ext_registry_mark(msgpack_packer_ext_registry_t* pkrg); void msgpack_packer_ext_registry_dup(msgpack_packer_ext_registry_t* src, msgpack_packer_ext_registry_t* dst); VALUE msgpack_packer_ext_registry_put(msgpack_packer_ext_registry_t* pkrg, VALUE ext_module, int ext_type, VALUE proc, VALUE arg); static int msgpack_packer_ext_find_superclass(VALUE key, VALUE value, VALUE arg) { VALUE *args = (VALUE *) arg; if(key == Qundef) { return ST_CONTINUE; } if(rb_class_inherited_p(args[0], key) == Qtrue) { args[1] = key; return ST_STOP; } return ST_CONTINUE; } static inline VALUE msgpack_packer_ext_registry_fetch(msgpack_packer_ext_registry_t* pkrg, VALUE lookup_class, int* ext_type_result) { // fetch lookup_class from hash, which is a hash to register classes VALUE type = rb_hash_lookup(pkrg->hash, lookup_class); if(type != Qnil) { *ext_type_result = FIX2INT(rb_ary_entry(type, 0)); return rb_ary_entry(type, 1); } // fetch lookup_class from cache, which stores results of searching ancestors from pkrg->hash VALUE type_inht = rb_hash_lookup(pkrg->cache, lookup_class); if(type_inht != Qnil) { *ext_type_result = FIX2INT(rb_ary_entry(type_inht, 0)); return rb_ary_entry(type_inht, 1); } return Qnil; } static inline VALUE msgpack_packer_ext_registry_lookup(msgpack_packer_ext_registry_t* pkrg, VALUE instance, int* ext_type_result) { VALUE lookup_class; VALUE type; /* * 1. check whether singleton_class of this instance is registered (or resolved in past) or not. * * Objects of type Integer (Fixnum, Bignum), Float, Symbol and frozen * String have no singleton class and raise a TypeError when trying to get * it. See implementation of #singleton_class in ruby's source code: * VALUE rb_singleton_class(VALUE obj); * * Since all but symbols are already filtered out when reaching this code * only symbols are checked here. */ if (!SYMBOL_P(instance)) { lookup_class = rb_singleton_class(instance); type = msgpack_packer_ext_registry_fetch(pkrg, lookup_class, ext_type_result); if(type != Qnil) { return type; } } /* * 2. check the class of instance is registered (or resolved in past) or not. */ type = msgpack_packer_ext_registry_fetch(pkrg, rb_obj_class(instance), ext_type_result); if(type != Qnil) { return type; } /* * 3. check all keys whether it is an ancestor of lookup_class, or not */ VALUE args[2]; args[0] = lookup_class; args[1] = Qnil; rb_hash_foreach(pkrg->hash, msgpack_packer_ext_find_superclass, (VALUE) args); VALUE superclass = args[1]; if(superclass != Qnil) { VALUE superclass_type = rb_hash_lookup(pkrg->hash, superclass); rb_hash_aset(pkrg->cache, lookup_class, superclass_type); *ext_type_result = FIX2INT(rb_ary_entry(superclass_type, 0)); return rb_ary_entry(superclass_type, 1); } return Qnil; } #endif