summaryrefslogtreecommitdiff
path: root/ext/openssl
diff options
context:
space:
mode:
authoremboss <emboss@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2011-05-22 00:01:06 +0000
committeremboss <emboss@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2011-05-22 00:01:06 +0000
commitdb874053457803ef694145c36249ad42aae5a49f (patch)
tree2fcb96c4eb3afb15cf4b69bb616f81da20d17917 /ext/openssl
parent3c25546ba288ef859ded3f9737464de3331107a1 (diff)
* ext/openssl/ossl_asn1.c: Default tag lookup in constant time via hash
instead of previous linear algorithm. [Ruby 1.9 - Feature #4309][ruby-core:34813] git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@31680 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
Diffstat (limited to 'ext/openssl')
-rw-r--r--ext/openssl/ossl_asn1.c42
1 files changed, 35 insertions, 7 deletions
diff --git a/ext/openssl/ossl_asn1.c b/ext/openssl/ossl_asn1.c
index 07746197ae..5ad1ffb0e0 100644
--- a/ext/openssl/ossl_asn1.c
+++ b/ext/openssl/ossl_asn1.c
@@ -489,6 +489,8 @@ static ossl_asn1_info_t ossl_asn1_info[] = {
int ossl_asn1_info_size = (sizeof(ossl_asn1_info)/sizeof(ossl_asn1_info[0]));
+static VALUE class_tag_map;
+
static int ossl_asn1_default_tag(VALUE obj);
ASN1_TYPE*
@@ -570,13 +572,13 @@ ossl_asn1_get_asn1type(VALUE obj)
static int
ossl_asn1_default_tag(VALUE obj)
{
- int i;
-
- for(i = 0; i < ossl_asn1_info_size; i++){
- if(ossl_asn1_info[i].klass &&
- rb_obj_is_kind_of(obj, *ossl_asn1_info[i].klass)){
- return i;
- }
+ VALUE tmp_class = CLASS_OF(obj);
+ while (tmp_class) {
+ VALUE tag = rb_hash_lookup(class_tag_map, tmp_class);
+ if (tag != Qnil) {
+ return NUM2INT(tag);
+ }
+ tmp_class = RCLASS_SUPER(tmp_class);
}
ossl_raise(eASN1Error, "universal tag for %s not found",
rb_class2name(CLASS_OF(obj)));
@@ -1765,4 +1767,30 @@ do{\
rb_attr(cASN1BitString, rb_intern("unused_bits"), 1, 1, 0);
rb_define_method(cASN1EndOfContent, "initialize", ossl_asn1eoc_initialize, 0);
+
+ class_tag_map = rb_hash_new();
+ rb_hash_aset(class_tag_map, cASN1EndOfContent, INT2NUM(0));
+ rb_hash_aset(class_tag_map, cASN1Boolean, INT2NUM(1));
+ rb_hash_aset(class_tag_map, cASN1Integer, INT2NUM(2));
+ rb_hash_aset(class_tag_map, cASN1BitString, INT2NUM(3));
+ rb_hash_aset(class_tag_map, cASN1OctetString, INT2NUM(4));
+ rb_hash_aset(class_tag_map, cASN1Null, INT2NUM(5));
+ rb_hash_aset(class_tag_map, cASN1ObjectId, INT2NUM(6));
+ rb_hash_aset(class_tag_map, cASN1Enumerated, INT2NUM(10));
+ rb_hash_aset(class_tag_map, cASN1UTF8String, INT2NUM(12));
+ rb_hash_aset(class_tag_map, cASN1Sequence, INT2NUM(16));
+ rb_hash_aset(class_tag_map, cASN1Set, INT2NUM(17));
+ rb_hash_aset(class_tag_map, cASN1NumericString, INT2NUM(18));
+ rb_hash_aset(class_tag_map, cASN1PrintableString, INT2NUM(19));
+ rb_hash_aset(class_tag_map, cASN1T61String, INT2NUM(20));
+ rb_hash_aset(class_tag_map, cASN1VideotexString, INT2NUM(21));
+ rb_hash_aset(class_tag_map, cASN1IA5String, INT2NUM(22));
+ rb_hash_aset(class_tag_map, cASN1UTCTime, INT2NUM(23));
+ rb_hash_aset(class_tag_map, cASN1GeneralizedTime, INT2NUM(24));
+ rb_hash_aset(class_tag_map, cASN1GraphicString, INT2NUM(25));
+ rb_hash_aset(class_tag_map, cASN1ISO64String, INT2NUM(26));
+ rb_hash_aset(class_tag_map, cASN1GeneralString, INT2NUM(27));
+ rb_hash_aset(class_tag_map, cASN1UniversalString, INT2NUM(28));
+ rb_hash_aset(class_tag_map, cASN1BMPString, INT2NUM(30));
+ rb_global_variable(&class_tag_map);
}