blob: e7253dcf2bcc65e335ba12ace967754c17632fed [file] [log] [blame]
Kevin Clark916f3532009-03-20 04:21:39 +00001/**
2 * Licensed to the Apache Software Foundation (ASF) under one
3 * or more contributor license agreements. See the NOTICE file
4 * distributed with this work for additional information
5 * regarding copyright ownership. The ASF licenses this file
6 * to you under the Apache License, Version 2.0 (the
7 * "License"); you may not use this file except in compliance
8 * with the License. You may obtain a copy of the License at
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing,
13 * software distributed under the License is distributed on an
14 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15 * KIND, either express or implied. See the License for the
16 * specific language governing permissions and limitations
17 * under the License.
18 */
19
Bryan Duxburyc0166282009-02-02 00:48:17 +000020#include <ruby.h>
21#include <constants.h>
Jake Farrellb5a18a12012-10-09 01:10:43 +000022#include <bytes.h>
23#include <macros.h>
Bryan Duxburyc0166282009-02-02 00:48:17 +000024
25ID buf_ivar_id;
26ID index_ivar_id;
27
28ID slice_method_id;
29
30int GARBAGE_BUFFER_SIZE;
31
32#define GET_BUF(self) rb_ivar_get(self, buf_ivar_id)
33
Bryan Duxburyad0ad822011-06-28 18:46:03 +000034VALUE rb_thrift_memory_buffer_write(VALUE self, VALUE str);
35VALUE rb_thrift_memory_buffer_read(VALUE self, VALUE length_value);
36VALUE rb_thrift_memory_buffer_read_byte(VALUE self);
37VALUE rb_thrift_memory_buffer_read_into_buffer(VALUE self, VALUE buffer_value, VALUE size_value);
38
Bryan Duxburyc0166282009-02-02 00:48:17 +000039VALUE rb_thrift_memory_buffer_write(VALUE self, VALUE str) {
40 VALUE buf = GET_BUF(self);
Jake Farrellb5a18a12012-10-09 01:10:43 +000041 str = force_binary_encoding(str);
Bryan Duxburye3ab50d2009-03-25 21:06:53 +000042 rb_str_buf_cat(buf, RSTRING_PTR(str), RSTRING_LEN(str));
Bryan Duxburyc0166282009-02-02 00:48:17 +000043 return Qnil;
44}
45
46VALUE rb_thrift_memory_buffer_read(VALUE self, VALUE length_value) {
47 int length = FIX2INT(length_value);
48
49 VALUE index_value = rb_ivar_get(self, index_ivar_id);
50 int index = FIX2INT(index_value);
51
52 VALUE buf = GET_BUF(self);
53 VALUE data = rb_funcall(buf, slice_method_id, 2, index_value, length_value);
54
55 index += length;
Bryan Duxburye3ab50d2009-03-25 21:06:53 +000056 if (index > RSTRING_LEN(buf)) {
57 index = RSTRING_LEN(buf);
Bryan Duxburyc0166282009-02-02 00:48:17 +000058 }
59 if (index >= GARBAGE_BUFFER_SIZE) {
Bryan Duxburye3ab50d2009-03-25 21:06:53 +000060 rb_ivar_set(self, buf_ivar_id, rb_funcall(buf, slice_method_id, 2, INT2FIX(index), INT2FIX(RSTRING_LEN(buf) - 1)));
Bryan Duxburyc0166282009-02-02 00:48:17 +000061 index = 0;
62 }
Roger Meier472f9e12011-08-02 10:55:47 +000063 rb_ivar_set(self, index_ivar_id, INT2FIX(index));
Bryan Duxburyc0166282009-02-02 00:48:17 +000064
Bryan Duxbury6f6318a2009-09-01 23:18:34 +000065 if (RSTRING_LEN(data) < length) {
Bryan Duxbury293086c2009-07-31 18:53:14 +000066 rb_raise(rb_eEOFError, "Not enough bytes remain in memory buffer");
67 }
68
Bryan Duxburyc0166282009-02-02 00:48:17 +000069 return data;
70}
71
Bryan Duxburyad0ad822011-06-28 18:46:03 +000072VALUE rb_thrift_memory_buffer_read_byte(VALUE self) {
73 VALUE index_value = rb_ivar_get(self, index_ivar_id);
74 int index = FIX2INT(index_value);
75
76 VALUE buf = GET_BUF(self);
77 if (index >= RSTRING_LEN(buf)) {
78 rb_raise(rb_eEOFError, "Not enough bytes remain in memory buffer");
79 }
80 char byte = RSTRING_PTR(buf)[index++];
Bryan Duxburyad0ad822011-06-28 18:46:03 +000081
82 if (index >= GARBAGE_BUFFER_SIZE) {
83 rb_ivar_set(self, buf_ivar_id, rb_funcall(buf, slice_method_id, 2, INT2FIX(index), INT2FIX(RSTRING_LEN(buf) - 1)));
84 index = 0;
85 }
Roger Meier472f9e12011-08-02 10:55:47 +000086 rb_ivar_set(self, index_ivar_id, INT2FIX(index));
87
Bryan Duxburyad0ad822011-06-28 18:46:03 +000088 int result = (int) byte;
89 return INT2FIX(result);
90}
91
92VALUE rb_thrift_memory_buffer_read_into_buffer(VALUE self, VALUE buffer_value, VALUE size_value) {
93 int i = 0;
94 int size = FIX2INT(size_value);
95 int index;
96 VALUE buf = GET_BUF(self);
97
Bryan Duxbury6530f1d2012-06-29 00:21:19 +000098 index = FIX2INT(rb_ivar_get(self, index_ivar_id));
Bryan Duxburyad0ad822011-06-28 18:46:03 +000099 while (i < size) {
Bryan Duxburyad0ad822011-06-28 18:46:03 +0000100 if (index >= RSTRING_LEN(buf)) {
101 rb_raise(rb_eEOFError, "Not enough bytes remain in memory buffer");
102 }
103 char byte = RSTRING_PTR(buf)[index++];
Bryan Duxburyad0ad822011-06-28 18:46:03 +0000104
Bryan Duxburyad0ad822011-06-28 18:46:03 +0000105 if (i >= RSTRING_LEN(buffer_value)) {
106 rb_raise(rb_eIndexError, "index %d out of string", i);
107 }
108 ((char*)RSTRING_PTR(buffer_value))[i] = byte;
109 i++;
110 }
Bryan Duxbury6530f1d2012-06-29 00:21:19 +0000111
112 if (index >= GARBAGE_BUFFER_SIZE) {
113 rb_ivar_set(self, buf_ivar_id, rb_funcall(buf, slice_method_id, 2, INT2FIX(index), INT2FIX(RSTRING_LEN(buf) - 1)));
114 index = 0;
115 }
116 rb_ivar_set(self, index_ivar_id, INT2FIX(index));
117
Bryan Duxburyad0ad822011-06-28 18:46:03 +0000118 return INT2FIX(i);
119}
120
Bryan Duxburyc0166282009-02-02 00:48:17 +0000121void Init_memory_buffer() {
Bryan Duxburyd1d15422009-04-04 00:58:03 +0000122 VALUE thrift_memory_buffer_class = rb_const_get(thrift_module, rb_intern("MemoryBufferTransport"));
Bryan Duxburyc0166282009-02-02 00:48:17 +0000123 rb_define_method(thrift_memory_buffer_class, "write", rb_thrift_memory_buffer_write, 1);
124 rb_define_method(thrift_memory_buffer_class, "read", rb_thrift_memory_buffer_read, 1);
Bryan Duxburyad0ad822011-06-28 18:46:03 +0000125 rb_define_method(thrift_memory_buffer_class, "read_byte", rb_thrift_memory_buffer_read_byte, 0);
126 rb_define_method(thrift_memory_buffer_class, "read_into_buffer", rb_thrift_memory_buffer_read_into_buffer, 2);
Bryan Duxburyc0166282009-02-02 00:48:17 +0000127
128 buf_ivar_id = rb_intern("@buf");
129 index_ivar_id = rb_intern("@index");
130
131 slice_method_id = rb_intern("slice");
132
133 GARBAGE_BUFFER_SIZE = FIX2INT(rb_const_get(thrift_memory_buffer_class, rb_intern("GARBAGE_BUFFER_SIZE")));
Bryan Duxbury1e80d442009-02-03 18:16:54 +0000134}