Optimize ByteBuffer#to_s and Mongo::ObjectID#generate via C extensions.

This commit is contained in:
Sean Cribbs 2009-10-29 11:44:07 -04:00
parent 5a81cb70ee
commit cb4d94f541
3 changed files with 81 additions and 14 deletions

View File

@ -38,6 +38,8 @@
#include <assert.h>
#include <math.h>
#include <unistd.h>
#include <time.h>
#define INITIAL_BUFFER_SIZE 256
@ -49,6 +51,7 @@ static VALUE Code;
static VALUE RegexpOfHolding;
static VALUE OrderedHash;
static VALUE InvalidName;
static VALUE DigestMD5;
#if HAVE_RUBY_ENCODING_H
#include "ruby/encoding.h"
@ -737,8 +740,58 @@ static VALUE method_deserialize(VALUE self, VALUE bson) {
return elements_to_hash(buffer, remaining);
}
static VALUE fast_pack(VALUE self)
{
VALUE res;
long i;
char c;
res = rb_str_buf_new(0);
for (i = 0; i < RARRAY(self)->len; i++) {
c = FIX2LONG(RARRAY(self)->ptr[i]);
rb_str_buf_cat(res, &c, sizeof(char));
}
return res;
}
static VALUE objectid_generate(VALUE self)
{
VALUE ary, digest, hostname;
unsigned char *string;
unsigned long t, pid, inc;
int i;
string = ALLOC_N(unsigned char, 13);
hostname = rb_str_buf_new(64);
t = htonl(time(NULL));
MEMCPY(string, &t, char, 4);
gethostname(RSTRING(hostname)->ptr, 64);
RSTRING(hostname)->len = strlen(RSTRING(hostname)->ptr);
digest = rb_funcall(DigestMD5, rb_intern("hexdigest"), 1, hostname);
MEMCPY(string+4, RSTRING(digest)->ptr, unsigned char, 3);
pid = htonl(getpid());
MEMCPY(string+7, &pid, char, 2);
inc = htonl(FIX2ULONG(rb_funcall(self, rb_intern("get_inc"), 0)));
MEMCPY(string+9, ((unsigned char*)&inc + 1), unsigned char, 3);
ary = rb_ary_new2(12);
for(i = 0; i < 12; i++)
rb_ary_store(ary, i, INT2FIX((unsigned int)string[i]));
free(string);
return ary;
}
void Init_cbson() {
VALUE mongo, CBson;
VALUE mongo, CBson, Digest;
Time = rb_const_get(rb_cObject, rb_intern("Time"));
mongo = rb_const_get(rb_cObject, rb_intern("Mongo"));
@ -760,4 +813,12 @@ void Init_cbson() {
CBson = rb_define_module("CBson");
rb_define_module_function(CBson, "serialize", method_serialize, 2);
rb_define_module_function(CBson, "deserialize", method_deserialize, 1);
rb_require("digest/md5");
Digest = rb_const_get(rb_cObject, rb_intern("Digest"));
DigestMD5 = rb_const_get(Digest, rb_intern("MD5"));
rb_define_method(ObjectID, "generate", objectid_generate, 0);
rb_define_method(rb_cArray, "fast_pack", fast_pack, 0);
}

View File

@ -120,23 +120,27 @@ module Mongo
private
def generate
oid = ''
begin
require 'mongo_ext/cbson'
rescue LoadError
def generate
oid = ''
# 4 bytes current time
time = Time.new.to_i
oid += [time].pack("N")
# 4 bytes current time
time = Time.new.to_i
oid += [time].pack("N")
# 3 bytes machine
oid += Digest::MD5.digest(Socket.gethostname)[0, 3]
# 3 bytes machine
oid += Digest::MD5.digest(Socket.gethostname)[0, 3]
# 2 bytes pid
oid += [Process.pid % 0xFFFF].pack("n")
# 2 bytes pid
oid += [Process.pid % 0xFFFF].pack("n")
# 3 bytes inc
oid += [get_inc].pack("N")[1, 3]
# 3 bytes inc
oid += [get_inc].pack("N")[1, 3]
oid.unpack("C12")
oid.unpack("C12")
end
end
def get_inc

View File

@ -159,7 +159,9 @@ class ByteBuffer
end
def to_s
if @buf.respond_to? "pack"
if @buf.respond_to? :fast_pack
@buf.fast_pack
elsif @buf.respond_to? "pack"
@buf.pack("C*")
else
@buf