Tamar 0.7.3 → 0.7.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/HISTORY +21 -3
- data/Rakefile +3 -0
- data/Tamar.gemspec +27 -2
- data/VERSION +1 -1
- data/src/build/extconf.rb +2 -0
- data/src/rubyluabridge/.hg_archival.txt +2 -0
- data/src/rubyluabridge/.project +91 -0
- data/src/rubyluabridge/LICENSE +31 -0
- data/src/rubyluabridge/LUA_IN_RUBY +146 -0
- data/src/rubyluabridge/README +129 -0
- data/src/rubyluabridge/RUBY_IN_LUA +25 -0
- data/src/rubyluabridge/Rakefile +88 -0
- data/src/rubyluabridge/build/extconf_osx.sh +2 -0
- data/src/rubyluabridge/build/extconf_ubuntu.sh +2 -0
- data/src/rubyluabridge/debian/changelog +34 -0
- data/src/rubyluabridge/debian/compat +1 -0
- data/src/rubyluabridge/debian/control +29 -0
- data/src/rubyluabridge/debian/copyright +34 -0
- data/src/rubyluabridge/debian/libluabridge-ruby1.8.doc-base +11 -0
- data/src/rubyluabridge/debian/libluabridge-ruby1.8.docs +2 -0
- data/src/rubyluabridge/debian/libluabridge-ruby1.8.install +1 -0
- data/src/rubyluabridge/debian/rules +65 -0
- data/src/rubyluabridge/debian/watch +2 -0
- data/src/rubyluabridge/doc/jamis.rb +591 -0
- data/src/rubyluabridge/extconf.rb +48 -0
- data/src/rubyluabridge/rubyluabridge.cc +1623 -0
- data/src/rubyluabridge/rubyluabridge.h +81 -0
- data/src/rubyluabridge/tests/lua_in_ruby_test.rb +508 -0
- metadata +28 -4
@@ -0,0 +1,1623 @@
|
|
1
|
+
/*
|
2
|
+
* RubyLuaBridge
|
3
|
+
*
|
4
|
+
* Licensed under the BSD License:
|
5
|
+
*
|
6
|
+
* Copyright (c) 2007, Evan Wies
|
7
|
+
* All rights reserved.
|
8
|
+
*
|
9
|
+
* Redistribution and use in source and binary forms, with or without
|
10
|
+
* modification, are permitted provided that the following conditions are met:
|
11
|
+
* * Redistributions of source code must retain the above copyright
|
12
|
+
* notice, this list of conditions and the following disclaimer.
|
13
|
+
* * Redistributions in binary form must reproduce the above copyright
|
14
|
+
* notice, this list of conditions and the following disclaimer in the
|
15
|
+
* documentation and/or other materials provided with the distribution.
|
16
|
+
* * Neither the name of 'neomantra' nor the
|
17
|
+
* names of its contributors may be used to endorse or promote products
|
18
|
+
* derived from this software without specific prior written permission.
|
19
|
+
*
|
20
|
+
* THIS SOFTWARE IS PROVIDED BY Evan Wies ``AS IS'' AND ANY
|
21
|
+
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
22
|
+
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
23
|
+
* DISCLAIMED. IN NO EVENT SHALL Evan Wies BE LIABLE FOR ANY
|
24
|
+
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
25
|
+
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
26
|
+
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
27
|
+
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
28
|
+
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
29
|
+
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
30
|
+
*/
|
31
|
+
|
32
|
+
#include <assert.h>
|
33
|
+
extern "C" {
|
34
|
+
#include <lua.h>
|
35
|
+
#include <lualib.h>
|
36
|
+
#include <lauxlib.h>
|
37
|
+
}
|
38
|
+
#include "rubyluabridge.h"
|
39
|
+
#include "st.h"
|
40
|
+
|
41
|
+
// Debug Scaffolding
|
42
|
+
#ifdef RLB_DEBUG
|
43
|
+
#define RLB_DEBUG_PRINT(fmt, ...) printf(fmt, __VA_ARGS__)
|
44
|
+
#else
|
45
|
+
#define RLB_DEBUG_PRINT(fmt, ...)
|
46
|
+
#endif
|
47
|
+
|
48
|
+
|
49
|
+
/*****************************************************************************/
|
50
|
+
//
|
51
|
+
// Global Variables and Typedefs
|
52
|
+
//
|
53
|
+
/*****************************************************************************/
|
54
|
+
|
55
|
+
VALUE mLua;
|
56
|
+
VALUE cLua_State;
|
57
|
+
VALUE cLua_RefObject;
|
58
|
+
VALUE cLua_Table; // derives from Lua::RefObject
|
59
|
+
|
60
|
+
|
61
|
+
/// A place to stash error messages
|
62
|
+
#define RUBYLUABRIDGE_ERROR_BUFFER_SIZE 4096
|
63
|
+
char gszErrorBuffer[RUBYLUABRIDGE_ERROR_BUFFER_SIZE];
|
64
|
+
|
65
|
+
|
66
|
+
/// WE DON"T DO THIS YET BUT PROBABLY WILL
|
67
|
+
/// In general, I'm doing a little cheating.
|
68
|
+
/// Since cLua_State is just a wrapper around a lua_State*,
|
69
|
+
/// and always can always extract the lua_State
|
70
|
+
/// meaning, take advantage the lua_State is ALWAYS there...
|
71
|
+
///
|
72
|
+
|
73
|
+
|
74
|
+
|
75
|
+
/*****************************************************************************/
|
76
|
+
//
|
77
|
+
// Library Infrastructure
|
78
|
+
//
|
79
|
+
/*****************************************************************************/
|
80
|
+
|
81
|
+
// Fwd: Ruby Hash iterator, inserting elements to a table
|
82
|
+
static int _rhash_to_table_iter_func( VALUE rkey, VALUE rvalue, lua_State* L );
|
83
|
+
|
84
|
+
|
85
|
+
/// Copies the top of the stack to gszErrorBuffer,
|
86
|
+
/// pops the top of the stack, and returns the address to gszErrorBuffer.
|
87
|
+
/// This is useful for error handling functions.
|
88
|
+
char* pop_error_to_buffer( lua_State* L )
|
89
|
+
{
|
90
|
+
const char* szerr = lua_tostring( L, -1 );
|
91
|
+
strncpy( gszErrorBuffer, szerr, RUBYLUABRIDGE_ERROR_BUFFER_SIZE );
|
92
|
+
lua_pop(L, 1);
|
93
|
+
return gszErrorBuffer;
|
94
|
+
}
|
95
|
+
|
96
|
+
|
97
|
+
/// return whether the value at a given index is "callable",
|
98
|
+
/// meaning that we can try to invoke it like a method
|
99
|
+
int is_callable( lua_State* L, int idx )
|
100
|
+
{
|
101
|
+
// functions are callable
|
102
|
+
if ( lua_type(L, idx) == LUA_TFUNCTION )
|
103
|
+
return 1;
|
104
|
+
|
105
|
+
// check if it is supports the __call metamethod
|
106
|
+
if ( !lua_getmetatable(L, idx) ) // no metatable?
|
107
|
+
return 0;
|
108
|
+
|
109
|
+
lua_pushstring( L, "__call" );
|
110
|
+
lua_rawget(L, -2);
|
111
|
+
int callable = (lua_isnil(L, -1) == 0);
|
112
|
+
|
113
|
+
lua_pop( L, 2 ); // remove metatable and metafield
|
114
|
+
return callable;
|
115
|
+
}
|
116
|
+
|
117
|
+
|
118
|
+
/// return whether the value at a given index is "indexable",
|
119
|
+
/// that means will doing a table-like lookup not crash
|
120
|
+
int is_indexable( lua_State* L, int idx )
|
121
|
+
{
|
122
|
+
// tables are obviously indexable
|
123
|
+
if ( lua_type(L, idx) == LUA_TTABLE )
|
124
|
+
return 1;
|
125
|
+
|
126
|
+
// check if it is supports the __index metamethod
|
127
|
+
if ( !lua_getmetatable(L, idx) ) // no metatable?
|
128
|
+
return 0;
|
129
|
+
|
130
|
+
lua_pushstring( L, "__index" );
|
131
|
+
lua_rawget(L, -2);
|
132
|
+
int indexable = (lua_isnil(L, -1) == 0);
|
133
|
+
|
134
|
+
lua_pop( L, 2 ); // remove metatable and metafield
|
135
|
+
return indexable;
|
136
|
+
}
|
137
|
+
|
138
|
+
|
139
|
+
/// return whether the value at a given index is "new_indexable",
|
140
|
+
/// that means will doing a table-like lookup not crash
|
141
|
+
int is_new_indexable( lua_State* L, int idx )
|
142
|
+
{
|
143
|
+
// tables are obviously indexable
|
144
|
+
if ( lua_type(L, idx) == LUA_TTABLE )
|
145
|
+
return 1;
|
146
|
+
|
147
|
+
// check if it is supports the __index metamethod
|
148
|
+
if ( !lua_getmetatable(L, idx) ) // no metatable?
|
149
|
+
return 0;
|
150
|
+
|
151
|
+
lua_pushstring( L, "__newindex" );
|
152
|
+
lua_rawget(L, -2);
|
153
|
+
int new_indexable = (lua_isnil(L, -1) == 0);
|
154
|
+
|
155
|
+
lua_pop( L, 2 ); // remove metatable and metafield
|
156
|
+
return new_indexable;
|
157
|
+
}
|
158
|
+
|
159
|
+
|
160
|
+
/// marshals the given stack value to Ruby
|
161
|
+
/// leaves the stack unchanged
|
162
|
+
VALUE marshal_lua_to_ruby( VALUE Rstate, lua_State* L, int idx )
|
163
|
+
{
|
164
|
+
int ltype = lua_type( L, idx );
|
165
|
+
switch ( ltype )
|
166
|
+
{
|
167
|
+
// primitive types -> marshal directly
|
168
|
+
case LUA_TNONE:
|
169
|
+
case LUA_TNIL:
|
170
|
+
return Qnil;
|
171
|
+
case LUA_TBOOLEAN:
|
172
|
+
return lua_toboolean(L,idx) ? Qtrue : Qfalse;
|
173
|
+
case LUA_TNUMBER:
|
174
|
+
return rb_float_new( lua_tonumber(L,idx) );
|
175
|
+
case LUA_TSTRING:
|
176
|
+
{
|
177
|
+
size_t len = 0;
|
178
|
+
const char* str = lua_tolstring(L, idx, &len);
|
179
|
+
return rb_str_new( str, len );
|
180
|
+
}
|
181
|
+
|
182
|
+
// complex types
|
183
|
+
// these are stored in Lua::RefObjects
|
184
|
+
// except tables, which are stored in Lua::Table
|
185
|
+
case LUA_TLIGHTUSERDATA:
|
186
|
+
case LUA_TFUNCTION:
|
187
|
+
case LUA_TUSERDATA:
|
188
|
+
case LUA_TTHREAD:
|
189
|
+
case LUA_TTABLE:
|
190
|
+
{
|
191
|
+
// make a Lua reference for this
|
192
|
+
lua_pushvalue( L, idx );
|
193
|
+
int ref = luaL_ref( L, LUA_REGISTRYINDEX );
|
194
|
+
RLB_DEBUG_PRINT( "ref created: L:%p r:%d\n", L, ref);
|
195
|
+
// create a new lua_Ref object holding it
|
196
|
+
VALUE args[2] = { Rstate, INT2NUM(ref) };
|
197
|
+
VALUE res = rb_class_new_instance( 2, args,
|
198
|
+
(ltype == LUA_TTABLE) ? cLua_Table : cLua_RefObject );
|
199
|
+
return res;
|
200
|
+
}
|
201
|
+
default:
|
202
|
+
return Qnil;
|
203
|
+
}
|
204
|
+
}
|
205
|
+
|
206
|
+
|
207
|
+
/// marshals the given Ruby value to Lua, leaving it on the top
|
208
|
+
/// returns a non-zero error code on failure
|
209
|
+
int marshal_ruby_to_lua_top( lua_State* L, VALUE val )
|
210
|
+
{
|
211
|
+
int rtype = TYPE(val);
|
212
|
+
switch ( rtype )
|
213
|
+
{
|
214
|
+
case T_NONE: lua_pushnil( L ); break;
|
215
|
+
case T_NIL: lua_pushnil( L ); break;
|
216
|
+
case T_TRUE: lua_pushboolean( L, 1 ); break;
|
217
|
+
case T_FALSE: lua_pushboolean( L, 0 ); break;
|
218
|
+
case T_FIXNUM: lua_pushnumber( L, FIX2INT(val) ); break;
|
219
|
+
case T_BIGNUM: lua_pushnumber( L, NUM2DBL(val) ); break;
|
220
|
+
case T_FLOAT:
|
221
|
+
lua_pushnumber( L, (lua_Number)RFLOAT(val)->value );
|
222
|
+
break;
|
223
|
+
case T_STRING:
|
224
|
+
lua_pushlstring( L, RSTRING(val)->ptr, RSTRING(val)->len );
|
225
|
+
break;
|
226
|
+
case T_SYMBOL:
|
227
|
+
lua_pushstring( L, rb_id2name( SYM2ID(val) ) );
|
228
|
+
break;
|
229
|
+
// Hash becomes a table
|
230
|
+
case T_HASH:
|
231
|
+
{
|
232
|
+
lua_newtable( L );
|
233
|
+
rb_hash_foreach( val, (int (*)(ANYARGS))_rhash_to_table_iter_func, VALUE(L) );
|
234
|
+
break;
|
235
|
+
}
|
236
|
+
// Array becomes a table-array.
|
237
|
+
// note in Lua, the first index is 1, whereas in Ruby the first index is 0
|
238
|
+
case T_ARRAY:
|
239
|
+
{
|
240
|
+
long i;
|
241
|
+
VALUE array = val;
|
242
|
+
lua_newtable( L );
|
243
|
+
for ( i = 0; i < RARRAY(array)->len; i++ )
|
244
|
+
{
|
245
|
+
marshal_ruby_to_lua_top( L, RARRAY(array)->ptr[i] );
|
246
|
+
lua_rawseti( L, -2, i+1 ); // Lua is 1-based
|
247
|
+
}
|
248
|
+
break;
|
249
|
+
}
|
250
|
+
case T_OBJECT:
|
251
|
+
{
|
252
|
+
// if it's a Lua::RefObject, put it on the stack
|
253
|
+
if ( rb_obj_is_kind_of(val, cLua_RefObject) == Qtrue )
|
254
|
+
{
|
255
|
+
rlua_RefObject* pRefObject;
|
256
|
+
Data_Get_Struct( val, rlua_RefObject, pRefObject );
|
257
|
+
if ( pRefObject->getState() != L )
|
258
|
+
{
|
259
|
+
// TODO: handle this better
|
260
|
+
rb_warning( "Marshalling Lua::RefObject between two different states. Pushing nil." );
|
261
|
+
break;
|
262
|
+
}
|
263
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
264
|
+
break;
|
265
|
+
}
|
266
|
+
// else, flow through to default...
|
267
|
+
}
|
268
|
+
default:
|
269
|
+
// just throw the Ruby object up as a light userdata
|
270
|
+
lua_pushlightuserdata( L, (void*)val );
|
271
|
+
break;
|
272
|
+
}
|
273
|
+
return 0;
|
274
|
+
}
|
275
|
+
|
276
|
+
|
277
|
+
// dispatches method_missing
|
278
|
+
//
|
279
|
+
// assumes that the Lua object (RefObject) is on the top of the stack
|
280
|
+
// leaves the stack clean and returns the corresponding value
|
281
|
+
//
|
282
|
+
// an = at the end of the key makes it an assignment
|
283
|
+
// an ! at the end of the key makes it a method call with self
|
284
|
+
// an _ at the end of the key forces it as a method call without self
|
285
|
+
// nothing at the end of the key will make a function call,
|
286
|
+
// unless there are no arguments in which case it simply returns obj[key]
|
287
|
+
//
|
288
|
+
VALUE rlua_method_missing_dispatch( lua_State* L, const char* key, VALUE Rstate, int argc, VALUE* argv )
|
289
|
+
{
|
290
|
+
int keylen = strlen( key );
|
291
|
+
|
292
|
+
// if method ends "=", then it's an assignment
|
293
|
+
if ( key[keylen-1] == '=' )
|
294
|
+
{
|
295
|
+
assert( argc >= 2 );
|
296
|
+
lua_pushlstring( L, key, keylen-1 ); // strip off =
|
297
|
+
marshal_ruby_to_lua_top( L, argv[1] );
|
298
|
+
lua_settable( L, -3 );
|
299
|
+
|
300
|
+
lua_pop( L, 1 );
|
301
|
+
return argv[1];
|
302
|
+
}
|
303
|
+
|
304
|
+
//
|
305
|
+
// otherwise read the index and return the value or invoke the function
|
306
|
+
//
|
307
|
+
|
308
|
+
// determine any special treatment and retrieve the value
|
309
|
+
int pushself = (key[keylen-1] == '!');
|
310
|
+
int forcecall = (key[keylen-1] == '_'); // keylen-1 to strip ! or _
|
311
|
+
lua_pushlstring( L, key, (pushself || forcecall) ? keylen-1 : keylen );
|
312
|
+
lua_gettable( L, -2 );
|
313
|
+
|
314
|
+
// if there are no arguments (besides self) and the method doesn't ends with a special char
|
315
|
+
// and it is not a function, then just return the indexed value.
|
316
|
+
if ( !pushself && !forcecall && argc == 1 && (lua_type(L,-1) != LUA_TFUNCTION) )
|
317
|
+
{
|
318
|
+
// marshal the result to Ruby
|
319
|
+
VALUE result = marshal_lua_to_ruby( Rstate, L, -1 );
|
320
|
+
lua_pop( L, 2 );
|
321
|
+
return result;
|
322
|
+
}
|
323
|
+
|
324
|
+
//
|
325
|
+
// otherwise, we're gonna pcall the indexed value
|
326
|
+
//
|
327
|
+
|
328
|
+
// make sure it is callable
|
329
|
+
if ( !is_callable(L,-1) )
|
330
|
+
{
|
331
|
+
int ltype = lua_type( L, -1 );
|
332
|
+
lua_pop( L, 2 );
|
333
|
+
rb_raise( rb_eRuntimeError,
|
334
|
+
"Value is not callable (not a function and no __call metamethod), ltype: %d, key: %s",
|
335
|
+
ltype, key );
|
336
|
+
}
|
337
|
+
|
338
|
+
// push the arguments on the stack
|
339
|
+
int args_bottom = lua_gettop(L) - 1;
|
340
|
+
if ( pushself )
|
341
|
+
lua_pushvalue( L, -2 ); // push self as arg with method!
|
342
|
+
int i;
|
343
|
+
for ( i = 1; i < argc; ++i )
|
344
|
+
marshal_ruby_to_lua_top( L, argv[i] );
|
345
|
+
|
346
|
+
// pcall and handle errors
|
347
|
+
int err = lua_pcall( L, (pushself ? argc : argc-1), LUA_MULTRET, 0 );
|
348
|
+
if ( err == LUA_ERRRUN ) {
|
349
|
+
lua_remove( L, -2 );
|
350
|
+
rb_raise( rb_eRuntimeError, pop_error_to_buffer(L) );
|
351
|
+
} else if ( err == LUA_ERRMEM ) {
|
352
|
+
lua_remove( L, -2 );
|
353
|
+
rb_raise( rb_eNoMemError, pop_error_to_buffer(L) );
|
354
|
+
} else if ( err == LUA_ERRERR ) {
|
355
|
+
lua_remove( L, -2 );
|
356
|
+
rb_raise( rb_eFatal, pop_error_to_buffer(L) );
|
357
|
+
}
|
358
|
+
|
359
|
+
// if there is one result, return that alone
|
360
|
+
// otherwise put them in an Array
|
361
|
+
int args_top = lua_gettop(L);
|
362
|
+
int nres = args_top - args_bottom;
|
363
|
+
if ( nres == 1 )
|
364
|
+
{
|
365
|
+
// marshal the result to Ruby
|
366
|
+
VALUE result = marshal_lua_to_ruby( Rstate, L, -1 );
|
367
|
+
lua_pop( L, 2 );
|
368
|
+
return result;
|
369
|
+
}
|
370
|
+
else
|
371
|
+
{
|
372
|
+
int li, ri;
|
373
|
+
VALUE ary_result = rb_ary_new2( nres );
|
374
|
+
for ( li = args_bottom+1, ri = 0; li <= args_top; ++li, ++ri )
|
375
|
+
rb_ary_store( ary_result, ri, marshal_lua_to_ruby(Rstate, L, li) );
|
376
|
+
lua_pop( L, 1+nres );
|
377
|
+
return ary_result;
|
378
|
+
}
|
379
|
+
}
|
380
|
+
|
381
|
+
|
382
|
+
/// Ruby Hash iterator, mapping pairs into a table
|
383
|
+
static int _rhash_to_table_iter_func( VALUE rkey, VALUE rvalue, lua_State* L )
|
384
|
+
{
|
385
|
+
marshal_ruby_to_lua_top( L, rkey );
|
386
|
+
marshal_ruby_to_lua_top( L, rvalue );
|
387
|
+
lua_settable( L, -3 );
|
388
|
+
return ST_CONTINUE;
|
389
|
+
}
|
390
|
+
|
391
|
+
|
392
|
+
// copied from:
|
393
|
+
// http://www.lua.org/source/5.1/lbaselib.c.html#luaB_tostring
|
394
|
+
static int rluaB_tostring(lua_State *L)
|
395
|
+
{
|
396
|
+
luaL_checkany(L, 1);
|
397
|
+
if (luaL_callmeta(L, 1, "__tostring")) /* is there a metafield? */
|
398
|
+
return 1; /* use its value */
|
399
|
+
switch (lua_type(L, 1))
|
400
|
+
{
|
401
|
+
case LUA_TNUMBER:
|
402
|
+
lua_pushstring(L, lua_tostring(L, 1));
|
403
|
+
break;
|
404
|
+
case LUA_TSTRING:
|
405
|
+
lua_pushvalue(L, 1);
|
406
|
+
break;
|
407
|
+
case LUA_TBOOLEAN:
|
408
|
+
lua_pushstring(L, (lua_toboolean(L, 1) ? "true" : "false"));
|
409
|
+
break;
|
410
|
+
case LUA_TNIL:
|
411
|
+
lua_pushliteral(L, "nil");
|
412
|
+
break;
|
413
|
+
default:
|
414
|
+
lua_pushfstring(L, "%s: %p", luaL_typename(L, 1), lua_topointer(L, 1));
|
415
|
+
break;
|
416
|
+
}
|
417
|
+
return 1;
|
418
|
+
}
|
419
|
+
|
420
|
+
|
421
|
+
// mapping of Lua Standard Library names to their registration functions
|
422
|
+
static const luaL_Reg rubylua_lualibs[] = {
|
423
|
+
{"", luaopen_base},
|
424
|
+
{LUA_LOADLIBNAME, luaopen_package},
|
425
|
+
{LUA_TABLIBNAME, luaopen_table},
|
426
|
+
{LUA_IOLIBNAME, luaopen_io},
|
427
|
+
{LUA_OSLIBNAME, luaopen_os},
|
428
|
+
{LUA_STRLIBNAME, luaopen_string},
|
429
|
+
{LUA_MATHLIBNAME, luaopen_math},
|
430
|
+
{LUA_DBLIBNAME, luaopen_debug},
|
431
|
+
{NULL, NULL}
|
432
|
+
};
|
433
|
+
|
434
|
+
|
435
|
+
/// loads the specified standard Lua library
|
436
|
+
void load_std_library_by_name( lua_State* L, const char* libname )
|
437
|
+
{
|
438
|
+
// special-case: base
|
439
|
+
const luaL_Reg* libreg = NULL;
|
440
|
+
if ( !strcmp(libname, "base") )
|
441
|
+
libreg = &rubylua_lualibs[0];
|
442
|
+
// otherwise search for it
|
443
|
+
else
|
444
|
+
{
|
445
|
+
for ( libreg = &rubylua_lualibs[1]; libreg->func; libreg++ )
|
446
|
+
if ( !strcmp(libname, libreg->name) )
|
447
|
+
break;
|
448
|
+
}
|
449
|
+
|
450
|
+
// register if found
|
451
|
+
if ( libreg && libreg->func )
|
452
|
+
{
|
453
|
+
lua_pushcfunction( L, libreg->func );
|
454
|
+
lua_pushstring( L, libreg->name );
|
455
|
+
lua_call( L, 1, 0 );
|
456
|
+
}
|
457
|
+
}
|
458
|
+
|
459
|
+
|
460
|
+
|
461
|
+
/*****************************************************************************
|
462
|
+
|
463
|
+
Document-class: Lua::State
|
464
|
+
|
465
|
+
The Ruby representation of a lua_State.
|
466
|
+
|
467
|
+
*****************************************************************************/
|
468
|
+
|
469
|
+
// Forward declaration
|
470
|
+
static VALUE rlua_State_loadlibs( VALUE self, VALUE libs );
|
471
|
+
|
472
|
+
/* call-seq:
|
473
|
+
* Lua::State.new( options )
|
474
|
+
*
|
475
|
+
* Creates a new Lua::State.
|
476
|
+
*
|
477
|
+
* _options_ is a hash of options. If no _options_ are specified, the default is { :loadlibs => :all }.
|
478
|
+
*
|
479
|
+
* <b>loadlibs:</b>
|
480
|
+
* Invokes Lua::State.__loadlibs on the new Lua::State, passing the value of :loadlibs.
|
481
|
+
*
|
482
|
+
* Raises NoMemoryError if the state cannot be allocated, or ArgumentError if the value of :loadlibs
|
483
|
+
* is invalid.
|
484
|
+
*/
|
485
|
+
static VALUE rlua_State_initialize( int argc, VALUE* argv, VALUE self )
|
486
|
+
{
|
487
|
+
rlua_State* pRLState;
|
488
|
+
Data_Get_Struct( self, rlua_State, pRLState );
|
489
|
+
|
490
|
+
// create new Lua state
|
491
|
+
pRLState->Lstate.reset( luaL_newstate(), lua_close_deleter() );
|
492
|
+
if ( !pRLState->getState() )
|
493
|
+
rb_raise( rb_eNoMemError, "lua_State memory allocation failed" );
|
494
|
+
|
495
|
+
RLB_DEBUG_PRINT( "state init: ptr:%p L:%p\n", pRLState, pRLState->getState() );
|
496
|
+
|
497
|
+
// if there is no arguments (or nil first value), load all
|
498
|
+
if ( argc == 0 || NIL_P(argv[0]) ) {
|
499
|
+
luaL_openlibs( pRLState->getState() );
|
500
|
+
return self;
|
501
|
+
}
|
502
|
+
// otherwise, it has to be a hash
|
503
|
+
Check_Type( argv[0], T_HASH );
|
504
|
+
|
505
|
+
// process "loadlibs"
|
506
|
+
VALUE libs = rb_hash_aref( argv[0], ID2SYM(rb_intern("loadlibs")) );
|
507
|
+
rlua_State_loadlibs( self, libs ); // OK if nil
|
508
|
+
|
509
|
+
return self;
|
510
|
+
}
|
511
|
+
|
512
|
+
|
513
|
+
/// free the lua_State, create with lua_State_alloc
|
514
|
+
static void rlua_State_free( rlua_State* pRLState )
|
515
|
+
{
|
516
|
+
RLB_DEBUG_PRINT( "state free : ptr:%p L:%p\n", pRLState, pRLState->getState() );
|
517
|
+
}
|
518
|
+
|
519
|
+
|
520
|
+
/// allocate a new Lua::State
|
521
|
+
/// this actually performs the luaL_newstate allocation
|
522
|
+
static VALUE rlua_State_alloc( VALUE klass )
|
523
|
+
{
|
524
|
+
rlua_State* pRLState = new rlua_State();
|
525
|
+
if ( !pRLState )
|
526
|
+
rb_raise( rb_eNoMemError, "Out of memory when allocating rlua_State" );
|
527
|
+
RLB_DEBUG_PRINT( "state malloc: ptr:%p\n", pRLState );
|
528
|
+
// wrap it inside a Ruby object
|
529
|
+
VALUE obj = Data_Wrap_Struct( klass, NULL, rlua_State_free, pRLState );
|
530
|
+
return obj;
|
531
|
+
}
|
532
|
+
|
533
|
+
|
534
|
+
/* call-seq:
|
535
|
+
* Lua::State.__state -> Lua::State
|
536
|
+
*
|
537
|
+
* Returns this Lua::State itself.
|
538
|
+
*
|
539
|
+
* Introduced for parallelism with Lua::RefObject.__state.
|
540
|
+
*/
|
541
|
+
static VALUE rlua_State_state( VALUE self )
|
542
|
+
{
|
543
|
+
return self;
|
544
|
+
}
|
545
|
+
|
546
|
+
|
547
|
+
/* call-seq:
|
548
|
+
* Lua::State.__top -> int
|
549
|
+
*
|
550
|
+
* Return the absolute position of the top of the lua_State's stack.
|
551
|
+
*
|
552
|
+
* This is mainly for debugging/testing purposes.
|
553
|
+
*/
|
554
|
+
static VALUE rlua_State_top( VALUE self )
|
555
|
+
{
|
556
|
+
rlua_State* pRLState;
|
557
|
+
Data_Get_Struct( self, rlua_State, pRLState );
|
558
|
+
int top = lua_gettop( pRLState->getState() );
|
559
|
+
return INT2NUM( top );
|
560
|
+
}
|
561
|
+
|
562
|
+
|
563
|
+
/* call-seq:
|
564
|
+
* Lua::State.__globals -> Lua::Table
|
565
|
+
*
|
566
|
+
* Returns the globals table of this Lua::State.
|
567
|
+
* It is an instance of the Lua::Table class.
|
568
|
+
*/
|
569
|
+
static VALUE rlua_State_globals( VALUE self )
|
570
|
+
{
|
571
|
+
rlua_State* pRLState;
|
572
|
+
Data_Get_Struct( self, rlua_State, pRLState );
|
573
|
+
lua_State* L = pRLState->getState();
|
574
|
+
|
575
|
+
lua_pushvalue( L, LUA_GLOBALSINDEX );
|
576
|
+
VALUE result = marshal_lua_to_ruby( self, L, -1 );
|
577
|
+
lua_pop( L, 1 );
|
578
|
+
return result;
|
579
|
+
}
|
580
|
+
|
581
|
+
|
582
|
+
/* call-seq:
|
583
|
+
* Lua::State.__registry -> Lua::Table
|
584
|
+
*
|
585
|
+
* Returns the registry table of this Lua::State.
|
586
|
+
* It is an instance of the Lua::Table class.
|
587
|
+
*
|
588
|
+
* As the Lua Registry is intended for C extensions
|
589
|
+
* and the Lua reference system, be careful modifying
|
590
|
+
* values stored in this table.
|
591
|
+
*/
|
592
|
+
static VALUE rlua_State_registry( VALUE self )
|
593
|
+
{
|
594
|
+
rlua_State* pRLState;
|
595
|
+
Data_Get_Struct( self, rlua_State, pRLState );
|
596
|
+
lua_State* L = pRLState->getState();
|
597
|
+
|
598
|
+
lua_pushvalue( L, LUA_REGISTRYINDEX );
|
599
|
+
VALUE result = marshal_lua_to_ruby( self, L, -1 );
|
600
|
+
lua_pop( L, 1 );
|
601
|
+
return result;
|
602
|
+
}
|
603
|
+
|
604
|
+
|
605
|
+
/* call-seq:
|
606
|
+
* Lua::State.__loadlib( libs )
|
607
|
+
*
|
608
|
+
* Loads the specified Lua standard libraries into the Lua::State.
|
609
|
+
*
|
610
|
+
* If _libs_ is not specified, all libraries are loaded. Otherwise, if _libs_ is a symbol, that library
|
611
|
+
* is loaded. Special symbols are :all, which loads all libraries, and :none which loads no libraries.
|
612
|
+
* If _libs_ is an Array, all symbols in the Array are loaded (in the order specified); in this case,
|
613
|
+
* :all and :none are ignored; an empty Array will load no libraries. If none of the above fits,
|
614
|
+
* an ArgumentError is raised.
|
615
|
+
*
|
616
|
+
* Supported libraries are:
|
617
|
+
* - base
|
618
|
+
* - package
|
619
|
+
* - table
|
620
|
+
* - io
|
621
|
+
* - os
|
622
|
+
* - string
|
623
|
+
* - math
|
624
|
+
* - debug
|
625
|
+
*/
|
626
|
+
static VALUE rlua_State_loadlibs( VALUE self, VALUE libs )
|
627
|
+
{
|
628
|
+
rlua_State* pRLState;
|
629
|
+
Data_Get_Struct( self, rlua_State, pRLState );
|
630
|
+
lua_State* L = pRLState->getState();
|
631
|
+
|
632
|
+
// if it is empty or :all, load all
|
633
|
+
// if it is a symbol, load that the lib it matches
|
634
|
+
// if it is :none, load none
|
635
|
+
// if it is an array, load all its symbols (:all is ignored here)
|
636
|
+
// otherwise, load none
|
637
|
+
if ( NIL_P(libs) )
|
638
|
+
{
|
639
|
+
luaL_openlibs( L );
|
640
|
+
}
|
641
|
+
else if ( TYPE(libs) == T_SYMBOL )
|
642
|
+
{
|
643
|
+
const char* libname = rb_id2name( SYM2ID(libs) );
|
644
|
+
if ( !strcmp(libname, "all") )
|
645
|
+
luaL_openlibs( L );
|
646
|
+
else if ( !strcmp(libname, "none") )
|
647
|
+
{} // load none on :none
|
648
|
+
else
|
649
|
+
load_std_library_by_name( L, libname );
|
650
|
+
}
|
651
|
+
else if ( TYPE(libs) == T_ARRAY )
|
652
|
+
{
|
653
|
+
int i;
|
654
|
+
for ( i = 0; i < RARRAY(libs)->len; i++ )
|
655
|
+
{
|
656
|
+
VALUE entry = RARRAY(libs)->ptr[i];
|
657
|
+
if ( TYPE(entry) == T_SYMBOL )
|
658
|
+
{
|
659
|
+
const char* libname = rb_id2name( SYM2ID(entry) );
|
660
|
+
load_std_library_by_name( L, libname );
|
661
|
+
}
|
662
|
+
}
|
663
|
+
}
|
664
|
+
else
|
665
|
+
rb_raise( rb_eArgError, "loadlibs must be Nil, a Symbol, or an Array of symbols" );
|
666
|
+
|
667
|
+
return self;
|
668
|
+
}
|
669
|
+
|
670
|
+
|
671
|
+
/* call-seq:
|
672
|
+
* Lua::State.eval -> result
|
673
|
+
*
|
674
|
+
* Evaluates the passed string in the Lua::State.
|
675
|
+
*
|
676
|
+
* Returns the first value returned by the evaluation.
|
677
|
+
*/
|
678
|
+
static VALUE rlua_State_eval( VALUE self, VALUE str )
|
679
|
+
{
|
680
|
+
// verify and marshal Ruby args to C
|
681
|
+
rlua_State* pRLState;
|
682
|
+
Data_Get_Struct( self, rlua_State, pRLState );
|
683
|
+
SafeStringValue(str);
|
684
|
+
lua_State* L = pRLState->getState();
|
685
|
+
|
686
|
+
// process the string to a chunk
|
687
|
+
int err = luaL_loadbuffer( L, RSTRING(str)->ptr, RSTRING(str)->len, "Lua::State.eval" );
|
688
|
+
if ( err == LUA_ERRMEM )
|
689
|
+
rb_raise( rb_eNoMemError, pop_error_to_buffer(L) );
|
690
|
+
else if ( err == LUA_ERRSYNTAX )
|
691
|
+
rb_raise( rb_eSyntaxError, pop_error_to_buffer(L) );
|
692
|
+
|
693
|
+
// pcall the chunk, returning only a single argument
|
694
|
+
// TODO: error handler with stack traceback
|
695
|
+
// TODO: it would be nice to have it configurable whether to print the traceback
|
696
|
+
// TODO: hmmm... the err handler could even be in Ruby?
|
697
|
+
err = lua_pcall( L, 0, 1, 0 );
|
698
|
+
if ( err == LUA_ERRRUN )
|
699
|
+
rb_raise( rb_eRuntimeError, pop_error_to_buffer(L) );
|
700
|
+
else if ( err == LUA_ERRMEM )
|
701
|
+
rb_raise( rb_eNoMemError, pop_error_to_buffer(L) );
|
702
|
+
else if ( err == LUA_ERRERR )
|
703
|
+
rb_raise( rb_eFatal, pop_error_to_buffer(L) );
|
704
|
+
|
705
|
+
// marshal the result to Ruby
|
706
|
+
VALUE result = marshal_lua_to_ruby( self, L, -1 );
|
707
|
+
lua_pop( L, 1 );
|
708
|
+
return result;
|
709
|
+
}
|
710
|
+
|
711
|
+
|
712
|
+
/* call-seq:
|
713
|
+
* Lua::State.eval_mult -> Array
|
714
|
+
*
|
715
|
+
* Evaluates the passed string in the Lua::State.
|
716
|
+
*
|
717
|
+
* Returns the all the return values in a Ruby array (with the first result first).
|
718
|
+
* If there are no results, an empty array is returned.
|
719
|
+
*/
|
720
|
+
static VALUE rlua_State_eval_mult( VALUE self, VALUE str )
|
721
|
+
{
|
722
|
+
// verify and marshal Ruby args to C
|
723
|
+
rlua_State* pRLState;
|
724
|
+
Data_Get_Struct( self, rlua_State, pRLState );
|
725
|
+
SafeStringValue(str);
|
726
|
+
lua_State* L = pRLState->getState();
|
727
|
+
int args_bottom = lua_gettop(L);
|
728
|
+
|
729
|
+
// process the string to a chunk
|
730
|
+
int err = luaL_loadbuffer( L, RSTRING(str)->ptr, RSTRING(str)->len, "Lua::State.eval" );
|
731
|
+
if ( err == LUA_ERRMEM )
|
732
|
+
rb_raise( rb_eNoMemError, pop_error_to_buffer(L) );
|
733
|
+
else if ( err == LUA_ERRSYNTAX )
|
734
|
+
rb_raise( rb_eSyntaxError, pop_error_to_buffer(L) );
|
735
|
+
|
736
|
+
// pcall the chunk, returning only a single argument
|
737
|
+
// TODO: error handler with stack traceback
|
738
|
+
// TODO: it would be nice to have it configurable whether to print the traceback
|
739
|
+
// TODO: hmmm... the err handler could even be in Ruby?
|
740
|
+
err = lua_pcall( L, 0, LUA_MULTRET, 0 );
|
741
|
+
if ( err == LUA_ERRRUN )
|
742
|
+
rb_raise( rb_eRuntimeError, pop_error_to_buffer(L) );
|
743
|
+
else if ( err == LUA_ERRMEM )
|
744
|
+
rb_raise( rb_eNoMemError, pop_error_to_buffer(L) );
|
745
|
+
else if ( err == LUA_ERRERR )
|
746
|
+
rb_raise( rb_eFatal, pop_error_to_buffer(L) );
|
747
|
+
|
748
|
+
// marshal the result to Ruby
|
749
|
+
int args_top = lua_gettop(L);
|
750
|
+
int nres = args_top - args_bottom;
|
751
|
+
int li, ri;
|
752
|
+
VALUE ary_res = rb_ary_new2( nres );
|
753
|
+
for ( li = args_bottom+1, ri = 0; li <= args_top; ++li, ++ri )
|
754
|
+
rb_ary_store( ary_res, ri, marshal_lua_to_ruby(self, L, li) );
|
755
|
+
lua_pop( L, nres );
|
756
|
+
return ary_res;
|
757
|
+
}
|
758
|
+
|
759
|
+
|
760
|
+
/* call-seq:
|
761
|
+
* Lua::State.method_missing -> result
|
762
|
+
*
|
763
|
+
* This method is called by Ruby when it sees an Object can't handle a message.
|
764
|
+
* We use it to dispatch to Lua, attempting a lookup of that value in the Lua::State's global table.
|
765
|
+
*
|
766
|
+
* If the method name has an '<tt>=</tt>' at the end, it is treated as an assignment,
|
767
|
+
* in which case it assigns the first value. It returns that value for chaining.
|
768
|
+
*
|
769
|
+
* The first argument is the symbol of the message name, the rest are its args.
|
770
|
+
*/
|
771
|
+
VALUE rlua_State_method_missing( int argc, VALUE* argv, VALUE self )
|
772
|
+
{
|
773
|
+
rlua_State* pRLstate;
|
774
|
+
Data_Get_Struct( self, rlua_State, pRLstate );
|
775
|
+
lua_State* L = pRLstate->getState();
|
776
|
+
|
777
|
+
Check_Type( argv[0], T_SYMBOL );
|
778
|
+
ID methodid = SYM2ID( argv[0] );
|
779
|
+
const char* key = rb_id2name( methodid );
|
780
|
+
|
781
|
+
lua_pushvalue( L, LUA_GLOBALSINDEX );
|
782
|
+
return rlua_method_missing_dispatch( L, key, self, argc, argv );
|
783
|
+
}
|
784
|
+
|
785
|
+
|
786
|
+
/* call-seq:
|
787
|
+
* Lua::State[key] -> value
|
788
|
+
*
|
789
|
+
* Returns the value indexed at _key_ in the Lua::State's globals table.
|
790
|
+
*/
|
791
|
+
VALUE rlua_State_getindex( VALUE self, VALUE key )
|
792
|
+
{
|
793
|
+
rlua_State* pRLstate;
|
794
|
+
Data_Get_Struct( self, rlua_State, pRLstate );
|
795
|
+
lua_State* L = pRLstate->getState();
|
796
|
+
|
797
|
+
marshal_ruby_to_lua_top( L, key );
|
798
|
+
lua_gettable( L, LUA_GLOBALSINDEX );
|
799
|
+
|
800
|
+
// marshal the result to Ruby
|
801
|
+
VALUE result = marshal_lua_to_ruby( self, L, -1 );
|
802
|
+
lua_pop( L, 1 );
|
803
|
+
return result;
|
804
|
+
}
|
805
|
+
|
806
|
+
|
807
|
+
/* call-seq:
|
808
|
+
* Lua::State[key] = value -> value
|
809
|
+
*
|
810
|
+
* Assigns _value_ to be indexed at _key_ in the Lua::State's globals table.
|
811
|
+
* Returns the value for chaining.
|
812
|
+
*/
|
813
|
+
VALUE rlua_State_setindex( VALUE self, VALUE key, VALUE val )
|
814
|
+
{
|
815
|
+
rlua_State* pRLstate;
|
816
|
+
Data_Get_Struct( self, rlua_State, pRLstate );
|
817
|
+
lua_State* L = pRLstate->getState();
|
818
|
+
|
819
|
+
marshal_ruby_to_lua_top( L, key );
|
820
|
+
marshal_ruby_to_lua_top( L, val );
|
821
|
+
lua_settable( L, LUA_GLOBALSINDEX );
|
822
|
+
|
823
|
+
return val; // return val for chaining
|
824
|
+
}
|
825
|
+
|
826
|
+
|
827
|
+
/* call-seq:
|
828
|
+
* Lua::State.new_table_at key -> Lua::Table
|
829
|
+
*
|
830
|
+
* Creates a new table at the given key. Returns the new table.
|
831
|
+
*/
|
832
|
+
VALUE rlua_State_new_table_at( VALUE self, VALUE key )
|
833
|
+
{
|
834
|
+
rlua_State* pRLstate;
|
835
|
+
Data_Get_Struct( self, rlua_State, pRLstate );
|
836
|
+
lua_State* L = pRLstate->getState();
|
837
|
+
|
838
|
+
marshal_ruby_to_lua_top( L, key );
|
839
|
+
lua_newtable( L );
|
840
|
+
|
841
|
+
VALUE result = marshal_lua_to_ruby( self, L, -1 );
|
842
|
+
lua_settable( L, LUA_GLOBALSINDEX );
|
843
|
+
|
844
|
+
return result;
|
845
|
+
}
|
846
|
+
|
847
|
+
|
848
|
+
/* call-seq:
|
849
|
+
* Lua::State.indexable? -> true
|
850
|
+
*
|
851
|
+
* Returns whether Lua:State is indexable (via __index), which it is.
|
852
|
+
* This is to provide consistency with Lua::RefObject interface.
|
853
|
+
*/
|
854
|
+
VALUE rlua_State_is_indexable( VALUE self )
|
855
|
+
{
|
856
|
+
return Qtrue;
|
857
|
+
}
|
858
|
+
|
859
|
+
|
860
|
+
/* call-seq:
|
861
|
+
* Lua::State.new_indexable? -> true
|
862
|
+
*
|
863
|
+
* Returns whether Lua:State can create new indices (via __newindex), which it can.
|
864
|
+
* This is to provide consistency with Lua::RefObject interface.
|
865
|
+
*/
|
866
|
+
VALUE rlua_State_is_new_indexable( VALUE self )
|
867
|
+
{
|
868
|
+
return Qtrue;
|
869
|
+
}
|
870
|
+
|
871
|
+
|
872
|
+
/* call-seq:
|
873
|
+
* Lua::State.callable? -> true
|
874
|
+
*
|
875
|
+
* Returns whether Lua:State is callable (like via __cal), which it is not..
|
876
|
+
* This is to provide consistency with Lua::RefObject interface.
|
877
|
+
*/
|
878
|
+
VALUE rlua_State_is_callable( VALUE self )
|
879
|
+
{
|
880
|
+
return Qfalse;
|
881
|
+
}
|
882
|
+
|
883
|
+
|
884
|
+
/*****************************************************************************
|
885
|
+
|
886
|
+
Document-class: Lua::RefObject
|
887
|
+
|
888
|
+
The Ruby representation of non-primitive objects in Lua.
|
889
|
+
|
890
|
+
*****************************************************************************/
|
891
|
+
|
892
|
+
/* call-seq:
|
893
|
+
* Lua::RefObject.new -> result
|
894
|
+
*
|
895
|
+
* Initializes a new Lua::RefObject.
|
896
|
+
*
|
897
|
+
* THIS METHOD IS NOT INTENDED TO BE CALLED BY CLIENTS.
|
898
|
+
*
|
899
|
+
* TODO: Can we enforce the hiding of this? Only Lua module members should access it.
|
900
|
+
*/
|
901
|
+
static VALUE rlua_RefObject_initialize( VALUE self, VALUE Rstate, VALUE RluaRef )
|
902
|
+
{
|
903
|
+
rlua_RefObject* pRef;
|
904
|
+
Data_Get_Struct( self, rlua_RefObject, pRef );
|
905
|
+
|
906
|
+
RLB_DEBUG_PRINT( "ref init: self:%d Rstate:%p TYPE(Rstate):%d RluaRef:%d TYPE(RluaRef):%d luaRef:%d\n",
|
907
|
+
(unsigned long)self, (unsigned long)Rstate, (unsigned long)TYPE(Rstate),
|
908
|
+
(unsigned long)RluaRef, (unsigned long)TYPE(RluaRef), NUM2INT(RluaRef) );
|
909
|
+
|
910
|
+
pRef->Rstate = Rstate;
|
911
|
+
pRef->Lref = NUM2INT(RluaRef);
|
912
|
+
|
913
|
+
rlua_State* pRState;
|
914
|
+
Data_Get_Struct( Rstate, rlua_State, pRState );
|
915
|
+
pRef->Lstate = pRState->Lstate;
|
916
|
+
|
917
|
+
return self;
|
918
|
+
}
|
919
|
+
|
920
|
+
|
921
|
+
/// free the Lua::RefObject, created with lua_RefObject_alloc
|
922
|
+
static void rlua_RefObject_free( rlua_RefObject* pRefObject )
|
923
|
+
{
|
924
|
+
RLB_DEBUG_PRINT( "ref free: ptr:%p ref:%d L:%p\n", pRefObject, pRefObject->Lref, pRefObject->getState() );
|
925
|
+
assert( pRefObject != NULL );
|
926
|
+
luaL_unref( pRefObject->getState(), LUA_REGISTRYINDEX, pRefObject->Lref );
|
927
|
+
|
928
|
+
delete pRefObject;
|
929
|
+
}
|
930
|
+
|
931
|
+
|
932
|
+
/// allocate a new Lua::RefObject
|
933
|
+
static VALUE rlua_RefObject_alloc( VALUE klass )
|
934
|
+
{
|
935
|
+
rlua_RefObject* pRef = new rlua_RefObject();
|
936
|
+
if ( !pRef )
|
937
|
+
rb_raise( rb_eNoMemError, "Out of memory when allocating rlua_RefObject" );
|
938
|
+
// pRef->Lstate = NULL;
|
939
|
+
pRef->Lref = LUA_NOREF;
|
940
|
+
pRef->Rstate = Qnil;
|
941
|
+
|
942
|
+
// wrap it inside a Ruby object
|
943
|
+
//rlua_RefObject_mark
|
944
|
+
VALUE obj = Data_Wrap_Struct( klass, NULL, rlua_RefObject_free, pRef );
|
945
|
+
return obj;
|
946
|
+
}
|
947
|
+
|
948
|
+
|
949
|
+
/* call-seq:
|
950
|
+
* Lua::RefObject.__state -> Lua::State
|
951
|
+
*
|
952
|
+
* Returns the Lua::State this Lua::RefObject belongs to.
|
953
|
+
*/
|
954
|
+
static VALUE rlua_RefObject_state( VALUE self )
|
955
|
+
{
|
956
|
+
rlua_RefObject* pRefObject;
|
957
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
958
|
+
|
959
|
+
return pRefObject->Rstate;
|
960
|
+
}
|
961
|
+
|
962
|
+
|
963
|
+
/* call-seq:
|
964
|
+
* Lua::RefObject.method_missing -> result
|
965
|
+
*
|
966
|
+
* This method is called by Ruby when it sees an Object can't handle a message.
|
967
|
+
* We use it to dispatch to Lua, attempting a lookup of that value in the Lua::RefObject.
|
968
|
+
*
|
969
|
+
* The first argument is the symbol of the message name, the rest are its args.
|
970
|
+
|
971
|
+
* If the method name has an <tt>=</tt> at the end, it is treated as an assignment,
|
972
|
+
* in which case it assigns the first value. It returns that value for chaining.
|
973
|
+
*
|
974
|
+
* If the method name has a '<tt>_</tt>' at the end, it is treated as an method invocation,
|
975
|
+
* passing itself as a parameter. This is to emulate the '<tt>:</tt>' token used in Lua.
|
976
|
+
*/
|
977
|
+
VALUE rlua_RefObject_method_missing( int argc, VALUE* argv, VALUE self )
|
978
|
+
{
|
979
|
+
rlua_RefObject* pRefObject;
|
980
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
981
|
+
lua_State* L = pRefObject->getState();
|
982
|
+
|
983
|
+
Check_Type( argv[0], T_SYMBOL );
|
984
|
+
ID methodid = SYM2ID( argv[0] );
|
985
|
+
const char* key = rb_id2name( methodid );
|
986
|
+
|
987
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
988
|
+
if ( !is_indexable(L, -1) )
|
989
|
+
{
|
990
|
+
lua_pop(L, 1);
|
991
|
+
rb_raise( rb_eRuntimeError, "Lua::RefObject not indexable, key='%s'", key );
|
992
|
+
}
|
993
|
+
|
994
|
+
return rlua_method_missing_dispatch( L, key, pRefObject->Rstate, argc, argv );
|
995
|
+
}
|
996
|
+
|
997
|
+
|
998
|
+
/* call-seq:
|
999
|
+
* Lua::RefObject.__length -> int
|
1000
|
+
*
|
1001
|
+
* Returns the 'length' of the RefObject.
|
1002
|
+
*
|
1003
|
+
* According to the {Lua manual}[http://www.lua.org/manual/5.1/manual.html#lua_objlen]:
|
1004
|
+
* Returns the "length" of the value at the given acceptable index:
|
1005
|
+
* for strings, this is the string length; for tables, this is the result
|
1006
|
+
* of the length operator ('#'); for userdata, this is the size of the
|
1007
|
+
* block of memory allocated for the userdata; for other values, it is 0.
|
1008
|
+
*/
|
1009
|
+
VALUE rlua_RefObject_length( VALUE self )
|
1010
|
+
{
|
1011
|
+
rlua_RefObject* pRefObject;
|
1012
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
1013
|
+
lua_State* L = pRefObject->getState();
|
1014
|
+
|
1015
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
1016
|
+
size_t len = lua_objlen(L, -1);
|
1017
|
+
lua_pop( L, 1 );
|
1018
|
+
|
1019
|
+
VALUE result = INT2NUM(len);
|
1020
|
+
return result;
|
1021
|
+
}
|
1022
|
+
|
1023
|
+
|
1024
|
+
/* call-seq:
|
1025
|
+
* Lua::RefObject.__type -> int
|
1026
|
+
*
|
1027
|
+
* Returns the type id of the underlying Lua object.
|
1028
|
+
*/
|
1029
|
+
VALUE rlua_RefObject_type( VALUE self )
|
1030
|
+
{
|
1031
|
+
rlua_RefObject* pRefObject;
|
1032
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
1033
|
+
lua_State* L = pRefObject->getState();
|
1034
|
+
|
1035
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
1036
|
+
size_t len = lua_type(L, -1);
|
1037
|
+
lua_pop( L, 1 );
|
1038
|
+
|
1039
|
+
VALUE result = INT2NUM(len);
|
1040
|
+
return result;
|
1041
|
+
}
|
1042
|
+
|
1043
|
+
|
1044
|
+
/* call-seq:
|
1045
|
+
* Lua::RefObject.__metatable -> Lua::Table
|
1046
|
+
*
|
1047
|
+
* Returns the metatable of the underlying Lua object.
|
1048
|
+
* Return nil if it has no metatable.
|
1049
|
+
*/
|
1050
|
+
VALUE rlua_RefObject_getmetatable( VALUE self )
|
1051
|
+
{
|
1052
|
+
rlua_RefObject* pRefObject;
|
1053
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
1054
|
+
lua_State* L = pRefObject->getState();
|
1055
|
+
|
1056
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
1057
|
+
|
1058
|
+
if (lua_getmetatable(L, -1) == 0)
|
1059
|
+
lua_pushnil( L );
|
1060
|
+
|
1061
|
+
VALUE result = marshal_lua_to_ruby( pRefObject->Rstate, L, -1 );
|
1062
|
+
lua_pop( L, 2 );
|
1063
|
+
return result;
|
1064
|
+
}
|
1065
|
+
|
1066
|
+
|
1067
|
+
/* call-seq:
|
1068
|
+
* Lua::RefObject.__metatable= Lua::Table -> Lua::Table
|
1069
|
+
*
|
1070
|
+
* Sets the metatable for this Lua::RefObject.
|
1071
|
+
* Returns the passed metatable.
|
1072
|
+
*/
|
1073
|
+
VALUE rlua_RefObject_setmetatable( VALUE self, VALUE mtable )
|
1074
|
+
{
|
1075
|
+
rlua_RefObject* pRefObject;
|
1076
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
1077
|
+
lua_State* L = pRefObject->getState();
|
1078
|
+
|
1079
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
1080
|
+
|
1081
|
+
marshal_ruby_to_lua_top( L, mtable );
|
1082
|
+
lua_setmetatable( L, -2 );
|
1083
|
+
|
1084
|
+
lua_pop( L, 1 );
|
1085
|
+
return mtable;
|
1086
|
+
}
|
1087
|
+
|
1088
|
+
|
1089
|
+
/* call-seq:
|
1090
|
+
* Lua::RefObject.[key] -> value
|
1091
|
+
*
|
1092
|
+
* Returns the value indexed at _key_ in the RefObject.
|
1093
|
+
*
|
1094
|
+
* Raises RuntimeError if the RefObject is not indexable.
|
1095
|
+
*/
|
1096
|
+
VALUE rlua_RefObject_getindex( VALUE self, VALUE key )
|
1097
|
+
{
|
1098
|
+
rlua_RefObject* pRefObject;
|
1099
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
1100
|
+
lua_State* L = pRefObject->getState();
|
1101
|
+
|
1102
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
1103
|
+
if ( !is_indexable(L, -1) )
|
1104
|
+
{
|
1105
|
+
lua_pop( L, 1 );
|
1106
|
+
rb_raise( rb_eRuntimeError, "(getindex) Lua::RefObject not indexable" );
|
1107
|
+
}
|
1108
|
+
|
1109
|
+
marshal_ruby_to_lua_top( L, key );
|
1110
|
+
lua_gettable( L, -2 );
|
1111
|
+
|
1112
|
+
// marshal the result to Ruby
|
1113
|
+
VALUE result = marshal_lua_to_ruby( pRefObject->Rstate, L, -1 );
|
1114
|
+
lua_pop( L, 2 );
|
1115
|
+
return result;
|
1116
|
+
}
|
1117
|
+
|
1118
|
+
|
1119
|
+
/* call-seq:
|
1120
|
+
* Lua::RefObject.[key] = value -> value
|
1121
|
+
*
|
1122
|
+
* Assigns _value_ to be indexed at _key_ in the RefObject.
|
1123
|
+
* Returns the value for chaining.
|
1124
|
+
*
|
1125
|
+
* Raises RuntimeError if the RefObject is not indexable.
|
1126
|
+
*/
|
1127
|
+
VALUE rlua_RefObject_setindex( VALUE self, VALUE key, VALUE val )
|
1128
|
+
{
|
1129
|
+
rlua_RefObject* pRefObject;
|
1130
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
1131
|
+
lua_State* L = pRefObject->getState();
|
1132
|
+
|
1133
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
1134
|
+
if ( !is_indexable(L, -1) )
|
1135
|
+
{
|
1136
|
+
lua_pop( L, 1 );
|
1137
|
+
rb_raise( rb_eRuntimeError, "(setindex) Lua::RefObject not indexable" );
|
1138
|
+
}
|
1139
|
+
|
1140
|
+
marshal_ruby_to_lua_top( L, key );
|
1141
|
+
marshal_ruby_to_lua_top( L, val );
|
1142
|
+
lua_settable( L, -3 );
|
1143
|
+
|
1144
|
+
lua_pop( L, 1 );
|
1145
|
+
return val; // return val for chaining
|
1146
|
+
}
|
1147
|
+
|
1148
|
+
|
1149
|
+
/* call-seq:
|
1150
|
+
* Lua::RefObject.new_table_at key -> Lua::Table
|
1151
|
+
*
|
1152
|
+
* Creates a new table at the given key. Returns the new table.
|
1153
|
+
*
|
1154
|
+
* Raises RuntimeError if the RefObject is not indexable.
|
1155
|
+
*/
|
1156
|
+
VALUE rlua_RefObject_new_table_at( VALUE self, VALUE key )
|
1157
|
+
{
|
1158
|
+
rlua_RefObject* pRefObject;
|
1159
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
1160
|
+
lua_State* L = pRefObject->getState();
|
1161
|
+
|
1162
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
1163
|
+
if ( !is_indexable(L, -1) )
|
1164
|
+
{
|
1165
|
+
lua_pop( L, 1 );
|
1166
|
+
rb_raise( rb_eRuntimeError, "(setindex) Lua::RefObject not indexable" );
|
1167
|
+
}
|
1168
|
+
|
1169
|
+
marshal_ruby_to_lua_top( L, key );
|
1170
|
+
lua_newtable( L );
|
1171
|
+
|
1172
|
+
VALUE result = marshal_lua_to_ruby( pRefObject->Rstate, L, -1 );
|
1173
|
+
lua_settable( L, -3 );
|
1174
|
+
|
1175
|
+
lua_pop( L, 1 );
|
1176
|
+
return result;
|
1177
|
+
}
|
1178
|
+
|
1179
|
+
|
1180
|
+
/* call-seq:
|
1181
|
+
* Lua::RefObject.to_s -> string
|
1182
|
+
*
|
1183
|
+
* Invokes the Lua function {tostring}[http://www.lua.org/manual/5.1/manual.html#pdf-tostring]
|
1184
|
+
* on the object and returns the resulting string.
|
1185
|
+
*/
|
1186
|
+
VALUE rlua_RefObject_to_s( VALUE self )
|
1187
|
+
{
|
1188
|
+
rlua_RefObject* pRefObject;
|
1189
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
1190
|
+
lua_State* L = pRefObject->getState();
|
1191
|
+
|
1192
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
1193
|
+
rluaB_tostring( L );
|
1194
|
+
|
1195
|
+
VALUE result = marshal_lua_to_ruby( pRefObject->Rstate, L, -1 );
|
1196
|
+
|
1197
|
+
lua_pop( L, 2 );
|
1198
|
+
return result;
|
1199
|
+
}
|
1200
|
+
|
1201
|
+
|
1202
|
+
/* call-seq:
|
1203
|
+
* Lua::RefObject.indexable? -> true
|
1204
|
+
*
|
1205
|
+
* Returns whether Lua:State is indexable (via __index), which it is.
|
1206
|
+
* This is to provide consistency with Lua::RefObject interface.
|
1207
|
+
*/
|
1208
|
+
VALUE rlua_RefObject_is_indexable( VALUE self )
|
1209
|
+
{
|
1210
|
+
rlua_RefObject* pRefObject;
|
1211
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
1212
|
+
lua_State* L = pRefObject->getState();
|
1213
|
+
|
1214
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
1215
|
+
VALUE result = is_indexable(L, -1) ? Qtrue : Qfalse;
|
1216
|
+
|
1217
|
+
lua_pop( L, 1 );
|
1218
|
+
return result;
|
1219
|
+
}
|
1220
|
+
|
1221
|
+
|
1222
|
+
/* call-seq:
|
1223
|
+
* Lua::RefObject.new_indexable? -> true
|
1224
|
+
*
|
1225
|
+
* Returns whether Lua:State can create new indices (via __newindex), which it can.
|
1226
|
+
* This is to provide consistency with Lua::RefObject interface.
|
1227
|
+
*/
|
1228
|
+
VALUE rlua_RefObject_is_new_indexable( VALUE self )
|
1229
|
+
{
|
1230
|
+
rlua_RefObject* pRefObject;
|
1231
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
1232
|
+
lua_State* L = pRefObject->getState();
|
1233
|
+
|
1234
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
1235
|
+
VALUE result = is_new_indexable(L, -1) ? Qtrue : Qfalse;
|
1236
|
+
|
1237
|
+
lua_pop( L, 1 );
|
1238
|
+
return result;
|
1239
|
+
}
|
1240
|
+
|
1241
|
+
|
1242
|
+
/* call-seq:
|
1243
|
+
* Lua::State.indexable? -> true
|
1244
|
+
*
|
1245
|
+
* Returns whether Lua:State is callable (like via __cal), which it is not..
|
1246
|
+
* This is to provide consistency with Lua::RefObject interface.
|
1247
|
+
*/
|
1248
|
+
VALUE rlua_RefObject_is_callable( VALUE self )
|
1249
|
+
{
|
1250
|
+
rlua_RefObject* pRefObject;
|
1251
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
1252
|
+
lua_State* L = pRefObject->getState();
|
1253
|
+
|
1254
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
1255
|
+
VALUE result = is_callable(L, -1) ? Qtrue : Qfalse;
|
1256
|
+
|
1257
|
+
lua_pop( L, 1 );
|
1258
|
+
return result;
|
1259
|
+
}
|
1260
|
+
|
1261
|
+
|
1262
|
+
|
1263
|
+
/*****************************************************************************
|
1264
|
+
|
1265
|
+
Document-class: Lua::Table
|
1266
|
+
|
1267
|
+
When Rua marshals a Lua table to Ruby, it instantiates it as a Lua::Table.
|
1268
|
+
|
1269
|
+
This gives it some extra methods that aren't available to a regular Lua::RefObject.
|
1270
|
+
|
1271
|
+
*****************************************************************************/
|
1272
|
+
|
1273
|
+
/////////////////////////////////
|
1274
|
+
///// LUA::TABLE CONVERSION
|
1275
|
+
|
1276
|
+
/* call-seq:
|
1277
|
+
* Lua::Table.to_array -> Array
|
1278
|
+
*
|
1279
|
+
* Returns a Ruby Array of the (first, dense) integer keys in the Table.
|
1280
|
+
*/
|
1281
|
+
VALUE rlua_Table_to_array( VALUE self )
|
1282
|
+
{
|
1283
|
+
rlua_RefObject* pRefObject;
|
1284
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
1285
|
+
lua_State* L = pRefObject->getState();
|
1286
|
+
|
1287
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
1288
|
+
int tablelen = lua_objlen( L, -1 );
|
1289
|
+
|
1290
|
+
VALUE array = rb_ary_new2( tablelen );
|
1291
|
+
int i;
|
1292
|
+
for ( i = 1; i <= tablelen; ++i )
|
1293
|
+
{
|
1294
|
+
lua_rawgeti( L, -1, i );
|
1295
|
+
VALUE rvalue = marshal_lua_to_ruby( pRefObject->Rstate, L, -1 );
|
1296
|
+
rb_ary_push( array, rvalue );
|
1297
|
+
lua_pop( L, 1 );
|
1298
|
+
}
|
1299
|
+
|
1300
|
+
lua_pop( L, 1 );
|
1301
|
+
return array;
|
1302
|
+
}
|
1303
|
+
|
1304
|
+
|
1305
|
+
/* call-seq:
|
1306
|
+
* Lua::Table.to_hash -> Hash
|
1307
|
+
*
|
1308
|
+
* Returns a Ruby Hash of all pairs in the table.
|
1309
|
+
*/
|
1310
|
+
VALUE rlua_Table_to_hash( VALUE self )
|
1311
|
+
{
|
1312
|
+
rlua_RefObject* pRefObject;
|
1313
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
1314
|
+
lua_State* L = pRefObject->getState();
|
1315
|
+
|
1316
|
+
VALUE hash = rb_hash_new();
|
1317
|
+
|
1318
|
+
// table is in the stack at index 't'
|
1319
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
1320
|
+
int table = lua_gettop( L );
|
1321
|
+
lua_pushnil( L ); // first key
|
1322
|
+
while ( lua_next(L, table) != 0 )
|
1323
|
+
{
|
1324
|
+
// uses 'key' (at index -2) and 'value' (at index -1)
|
1325
|
+
VALUE rvalue = marshal_lua_to_ruby( pRefObject->Rstate, L, -1 );
|
1326
|
+
VALUE rkey = marshal_lua_to_ruby( pRefObject->Rstate, L, -2 );
|
1327
|
+
rb_hash_aset( hash, rkey, rvalue );
|
1328
|
+
// removes 'value'; keeps 'key' for next iteration
|
1329
|
+
lua_pop( L, 1 );
|
1330
|
+
}
|
1331
|
+
|
1332
|
+
lua_pop( L, 1 );
|
1333
|
+
return hash;
|
1334
|
+
}
|
1335
|
+
|
1336
|
+
|
1337
|
+
/////////////////////////////////
|
1338
|
+
///// LUA::TABLE HASH ITERATION
|
1339
|
+
|
1340
|
+
/* call-seq:
|
1341
|
+
* Lua::Table.each { |key,value| block } -> Lua::Table
|
1342
|
+
* Lua::Table.each_pair { |key,value| block } -> Lua::Table
|
1343
|
+
*
|
1344
|
+
* Calls _block_ once for each key in _table_, passing the key and value as parameters.
|
1345
|
+
*
|
1346
|
+
* This goes over all pairs in the table.
|
1347
|
+
*/
|
1348
|
+
VALUE rlua_Table_each_pair( VALUE self )
|
1349
|
+
{
|
1350
|
+
rlua_RefObject* pRefObject;
|
1351
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
1352
|
+
lua_State* L = pRefObject->getState();
|
1353
|
+
|
1354
|
+
// table is in the stack at index 't'
|
1355
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
1356
|
+
int table = lua_gettop( L );
|
1357
|
+
lua_pushnil( L ); // first key
|
1358
|
+
while ( lua_next(L, table) != 0 )
|
1359
|
+
{
|
1360
|
+
// uses 'key' (at index -2) and 'value' (at index -1)
|
1361
|
+
VALUE rvalue = marshal_lua_to_ruby( pRefObject->Rstate, L, -1 );
|
1362
|
+
VALUE rkey = marshal_lua_to_ruby( pRefObject->Rstate, L, -2 );
|
1363
|
+
rb_yield_values( 2, rkey, rvalue );
|
1364
|
+
// removes 'value'; keeps 'key' for next iteration
|
1365
|
+
lua_pop( L, 1 );
|
1366
|
+
}
|
1367
|
+
|
1368
|
+
lua_pop( L, 1 );
|
1369
|
+
return self;
|
1370
|
+
}
|
1371
|
+
|
1372
|
+
|
1373
|
+
/* call-seq:
|
1374
|
+
* Lua::Table.each_key { |key| block } -> Lua::Table
|
1375
|
+
*
|
1376
|
+
* Calls _block_ once for each key in _table_, passing the key to the block as a parameter.
|
1377
|
+
*
|
1378
|
+
* This goes over all pairs in the table.
|
1379
|
+
*/
|
1380
|
+
VALUE rlua_Table_each_key( VALUE self )
|
1381
|
+
{
|
1382
|
+
rlua_RefObject* pRefObject;
|
1383
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
1384
|
+
lua_State* L = pRefObject->getState();
|
1385
|
+
|
1386
|
+
// table is in the stack at index 't'
|
1387
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
1388
|
+
int table = lua_gettop( L );
|
1389
|
+
lua_pushnil( L ); // first key
|
1390
|
+
while ( lua_next(L, table) != 0 )
|
1391
|
+
{
|
1392
|
+
// uses 'key' (at index -2) and 'value' (at index -1)
|
1393
|
+
VALUE rkey = marshal_lua_to_ruby( pRefObject->Rstate, L, -2 );
|
1394
|
+
rb_yield( rkey );
|
1395
|
+
// removes 'value'; keeps 'key' for next iteration
|
1396
|
+
lua_pop( L, 1 );
|
1397
|
+
}
|
1398
|
+
|
1399
|
+
lua_pop( L, 1 );
|
1400
|
+
return self;
|
1401
|
+
}
|
1402
|
+
|
1403
|
+
|
1404
|
+
/* call-seq:
|
1405
|
+
* Lua::Table.each_value { |value| block } -> Lua::Table
|
1406
|
+
*
|
1407
|
+
* Calls _block_ once for each key in _table_, passing the value to the block as a parameter.
|
1408
|
+
*
|
1409
|
+
* This goes over all pairs in the table.
|
1410
|
+
*/
|
1411
|
+
VALUE rlua_Table_each_value( VALUE self )
|
1412
|
+
{
|
1413
|
+
rlua_RefObject* pRefObject;
|
1414
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
1415
|
+
lua_State* L = pRefObject->getState();
|
1416
|
+
|
1417
|
+
// table is in the stack at index 't'
|
1418
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
1419
|
+
int table = lua_gettop( L );
|
1420
|
+
lua_pushnil( L ); // first key
|
1421
|
+
while ( lua_next(L, table) != 0 )
|
1422
|
+
{
|
1423
|
+
// uses 'key' (at index -2) and 'value' (at index -1)
|
1424
|
+
VALUE rvalue = marshal_lua_to_ruby( pRefObject->Rstate, L, -1 );
|
1425
|
+
rb_yield( rvalue );
|
1426
|
+
// removes 'value'; keeps 'key' for next iteration
|
1427
|
+
lua_pop( L, 1 );
|
1428
|
+
}
|
1429
|
+
|
1430
|
+
lua_pop( L, 1 );
|
1431
|
+
return self;
|
1432
|
+
}
|
1433
|
+
|
1434
|
+
|
1435
|
+
/////////////////////////
|
1436
|
+
///// ARRAY ITERATION
|
1437
|
+
|
1438
|
+
/* call-seq:
|
1439
|
+
* Lua::Table.each_ipair { |key,value| block } -> Lua::Table
|
1440
|
+
*
|
1441
|
+
* Calls _block_ once for each integer key in _table_, passing the key and value as parameters.
|
1442
|
+
*
|
1443
|
+
* This goes over all integer pairs in the table. Similar to ipairs(), this
|
1444
|
+
* only touches the first #table integers, thus treating the table like a dense array.
|
1445
|
+
*/
|
1446
|
+
VALUE rlua_Table_each_ipair( VALUE self )
|
1447
|
+
{
|
1448
|
+
rlua_RefObject* pRefObject;
|
1449
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
1450
|
+
lua_State* L = pRefObject->getState();
|
1451
|
+
|
1452
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
1453
|
+
int tablelen = lua_objlen( L, -1 );
|
1454
|
+
int i;
|
1455
|
+
for ( i = 1; i <= tablelen; ++i )
|
1456
|
+
{
|
1457
|
+
lua_rawgeti( L, -1, i );
|
1458
|
+
VALUE rvalue = marshal_lua_to_ruby( pRefObject->Rstate, L, -1 );
|
1459
|
+
rb_yield_values( 2, INT2NUM(i), rvalue );
|
1460
|
+
lua_pop( L, 1 );
|
1461
|
+
}
|
1462
|
+
|
1463
|
+
lua_pop( L, 1 );
|
1464
|
+
|
1465
|
+
return self;
|
1466
|
+
}
|
1467
|
+
|
1468
|
+
|
1469
|
+
/* call-seq:
|
1470
|
+
* Lua::Table.each_index { |key| block } -> Lua::Table
|
1471
|
+
* Lua::Table.each_ikey { |key| block } -> Lua::Table
|
1472
|
+
*
|
1473
|
+
* Calls _block_ once for each integer key in _table_, passing the key to the block as a parameter.
|
1474
|
+
*
|
1475
|
+
* This goes over all integer pairs in the table. Similar to ipairs(), this
|
1476
|
+
* only touches the first #table integers, thus treating the table like a dense array.
|
1477
|
+
*/
|
1478
|
+
VALUE rlua_Table_each_ikey( VALUE self )
|
1479
|
+
{
|
1480
|
+
rlua_RefObject* pRefObject;
|
1481
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
1482
|
+
lua_State* L = pRefObject->getState();
|
1483
|
+
|
1484
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
1485
|
+
int tablelen = lua_objlen( L, -1 );
|
1486
|
+
int i;
|
1487
|
+
for ( i = 1; i <= tablelen; ++i )
|
1488
|
+
{
|
1489
|
+
lua_rawgeti( L, -1, i );
|
1490
|
+
rb_yield( INT2NUM(i) );
|
1491
|
+
lua_pop( L, 1 );
|
1492
|
+
}
|
1493
|
+
|
1494
|
+
lua_pop( L, 1 );
|
1495
|
+
|
1496
|
+
return self;
|
1497
|
+
}
|
1498
|
+
|
1499
|
+
|
1500
|
+
/* call-seq:
|
1501
|
+
* Lua::Table.each_ivalue { |value| block } -> Lua::Table
|
1502
|
+
*
|
1503
|
+
* Calls _block_ once for each integer key in _table_, passing the value to the block as a parameter.
|
1504
|
+
*
|
1505
|
+
* This goes over all integer pairs in the table. Similar to ipairs(), this
|
1506
|
+
* only touches the first #table integers, thus treating the table like a dense array.
|
1507
|
+
*/
|
1508
|
+
VALUE rlua_Table_each_ivalue( VALUE self )
|
1509
|
+
{
|
1510
|
+
rlua_RefObject* pRefObject;
|
1511
|
+
Data_Get_Struct( self, rlua_RefObject, pRefObject );
|
1512
|
+
lua_State* L = pRefObject->getState();
|
1513
|
+
|
1514
|
+
lua_rawgeti( L, LUA_REGISTRYINDEX, pRefObject->Lref );
|
1515
|
+
int tablelen = lua_objlen( L, -1 );
|
1516
|
+
int i;
|
1517
|
+
for ( i = 1; i <= tablelen; ++i )
|
1518
|
+
{
|
1519
|
+
lua_rawgeti( L, -1, i );
|
1520
|
+
VALUE rvalue = marshal_lua_to_ruby( pRefObject->Rstate, L, -1 );
|
1521
|
+
rb_yield( rvalue );
|
1522
|
+
lua_pop( L, 1 );
|
1523
|
+
}
|
1524
|
+
|
1525
|
+
lua_pop( L, 1 );
|
1526
|
+
|
1527
|
+
return self;
|
1528
|
+
}
|
1529
|
+
|
1530
|
+
|
1531
|
+
|
1532
|
+
/*****************************************************************************/
|
1533
|
+
//
|
1534
|
+
// Extension Initialization
|
1535
|
+
//
|
1536
|
+
/*****************************************************************************/
|
1537
|
+
|
1538
|
+
extern "C" {
|
1539
|
+
/*
|
1540
|
+
Document-module: Lua
|
1541
|
+
|
1542
|
+
The module where all of the RubyLuaBridge components live.
|
1543
|
+
*/
|
1544
|
+
void Init_rubyluabridge()
|
1545
|
+
{
|
1546
|
+
// Lua module
|
1547
|
+
mLua = rb_define_module( "Lua" );
|
1548
|
+
rb_define_const( mLua, "BRIDGE_VERSION", rb_str_new2( RUBYLUABRIDGE_VERSION ) );
|
1549
|
+
rb_define_const( mLua, "BRIDGE_VERSION_NUM", INT2FIX( RUBYLUABRIDGE_VERSION_NUM ) );
|
1550
|
+
rb_define_const( mLua, "LUA_VERSION", rb_str_new2( LUA_VERSION ) );
|
1551
|
+
rb_define_const( mLua, "LUA_RELEASE", rb_str_new2( LUA_RELEASE ) );
|
1552
|
+
|
1553
|
+
rb_define_const( mLua, "TNONE", INT2FIX( LUA_TNONE ) );
|
1554
|
+
rb_define_const( mLua, "TNIL", INT2FIX( LUA_TNIL ) );
|
1555
|
+
rb_define_const( mLua, "TBOOLEAN", INT2FIX( LUA_TBOOLEAN ) );
|
1556
|
+
rb_define_const( mLua, "TLIGHTUSERDATA", INT2FIX( LUA_TLIGHTUSERDATA ) );
|
1557
|
+
rb_define_const( mLua, "TNUMBER", INT2FIX( LUA_TNUMBER ) );
|
1558
|
+
rb_define_const( mLua, "TSTRING", INT2FIX( LUA_TSTRING ) );
|
1559
|
+
rb_define_const( mLua, "TTABLE", INT2FIX( LUA_TTABLE ) );
|
1560
|
+
rb_define_const( mLua, "TFUNCTION", INT2FIX( LUA_TFUNCTION ) );
|
1561
|
+
rb_define_const( mLua, "TUSERDATA", INT2FIX( LUA_TUSERDATA ) );
|
1562
|
+
rb_define_const( mLua, "TTHREAD", INT2FIX( LUA_TTHREAD ) );
|
1563
|
+
|
1564
|
+
// Lua::State class
|
1565
|
+
cLua_State = rb_define_class_under( mLua, "State", rb_cObject );
|
1566
|
+
rb_define_alloc_func( cLua_State, rlua_State_alloc );
|
1567
|
+
rb_define_method( cLua_State, "initialize", RUBY_METHOD_FUNC(rlua_State_initialize), -1 );
|
1568
|
+
rb_define_method( cLua_State, "method_missing", RUBY_METHOD_FUNC(rlua_State_method_missing), -1 );
|
1569
|
+
rb_define_method( cLua_State, "__state", RUBY_METHOD_FUNC(rlua_State_state), 0 );
|
1570
|
+
rb_define_method( cLua_State, "__top", RUBY_METHOD_FUNC(rlua_State_top), 0 );
|
1571
|
+
rb_define_method( cLua_State, "__globals", RUBY_METHOD_FUNC(rlua_State_globals), 0 );
|
1572
|
+
rb_define_method( cLua_State, "__registry", RUBY_METHOD_FUNC(rlua_State_registry), 0 );
|
1573
|
+
rb_define_method( cLua_State, "__loadlibs", RUBY_METHOD_FUNC(rlua_State_loadlibs), 1 );
|
1574
|
+
rb_define_method( cLua_State, "new_table_at", RUBY_METHOD_FUNC(rlua_State_new_table_at), 1 );
|
1575
|
+
rb_define_method( cLua_State, "[]", RUBY_METHOD_FUNC(rlua_State_getindex), 1 );
|
1576
|
+
rb_define_method( cLua_State, "[]=", RUBY_METHOD_FUNC(rlua_State_setindex), 2 );
|
1577
|
+
rb_define_method( cLua_State, "eval", RUBY_METHOD_FUNC(rlua_State_eval), 1 );
|
1578
|
+
rb_define_method( cLua_State, "eval_mult", RUBY_METHOD_FUNC(rlua_State_eval_mult), 1 );
|
1579
|
+
rb_define_method( cLua_State, "callable?", RUBY_METHOD_FUNC(rlua_State_is_callable), 0 );
|
1580
|
+
rb_define_method( cLua_State, "indexable?", RUBY_METHOD_FUNC(rlua_State_is_indexable), 0 );
|
1581
|
+
rb_define_method( cLua_State, "new_indexable?", RUBY_METHOD_FUNC(rlua_State_is_new_indexable), 0 );
|
1582
|
+
// TODO: more methods!
|
1583
|
+
|
1584
|
+
// rb_define_method( cLua_State, "execute", lua_State_eval, 1 );
|
1585
|
+
// rb_define_method( cLua_State, "execute_mult", lua_State_eval_mult, 1 );
|
1586
|
+
|
1587
|
+
// Lua::RefObject class
|
1588
|
+
cLua_RefObject = rb_define_class_under( mLua, "RefObject", rb_cObject );
|
1589
|
+
rb_define_alloc_func( cLua_RefObject, rlua_RefObject_alloc );
|
1590
|
+
rb_define_method( cLua_RefObject, "initialize", RUBY_METHOD_FUNC(rlua_RefObject_initialize), 2 );
|
1591
|
+
rb_define_method( cLua_RefObject, "method_missing", RUBY_METHOD_FUNC(rlua_RefObject_method_missing), -1 );
|
1592
|
+
rb_define_method( cLua_RefObject, "__state", RUBY_METHOD_FUNC(rlua_RefObject_state), 0 );
|
1593
|
+
rb_define_method( cLua_RefObject, "__length", RUBY_METHOD_FUNC(rlua_RefObject_length), 0 );
|
1594
|
+
rb_define_method( cLua_RefObject, "__type", RUBY_METHOD_FUNC(rlua_RefObject_type), 0 );
|
1595
|
+
rb_define_method( cLua_RefObject, "__metatable", RUBY_METHOD_FUNC(rlua_RefObject_getmetatable), 0 );
|
1596
|
+
rb_define_method( cLua_RefObject, "__metatable=", RUBY_METHOD_FUNC(rlua_RefObject_setmetatable), 1 );
|
1597
|
+
rb_define_method( cLua_RefObject, "[]", RUBY_METHOD_FUNC(rlua_RefObject_getindex), 1 );
|
1598
|
+
rb_define_method( cLua_RefObject, "[]=", RUBY_METHOD_FUNC(rlua_RefObject_setindex), 2 );
|
1599
|
+
rb_define_method( cLua_RefObject, "new_table_at", RUBY_METHOD_FUNC(rlua_RefObject_new_table_at), 1 );
|
1600
|
+
rb_define_method( cLua_RefObject, "to_s", RUBY_METHOD_FUNC(rlua_RefObject_to_s), 0 );
|
1601
|
+
rb_define_method( cLua_RefObject, "method_missing", RUBY_METHOD_FUNC(rlua_RefObject_method_missing), -1 );
|
1602
|
+
rb_undef_method( cLua_RefObject, "type" );
|
1603
|
+
rb_undef_method( cLua_RefObject, "id" );
|
1604
|
+
rb_define_method( cLua_RefObject, "callable?", RUBY_METHOD_FUNC(rlua_RefObject_is_callable), 0 );
|
1605
|
+
rb_define_method( cLua_RefObject, "indexable?", RUBY_METHOD_FUNC(rlua_RefObject_is_indexable), 0 );
|
1606
|
+
rb_define_method( cLua_RefObject, "new_indexable?", RUBY_METHOD_FUNC(rlua_RefObject_is_new_indexable), 0 );
|
1607
|
+
|
1608
|
+
// Lua::Table class
|
1609
|
+
cLua_Table = rb_define_class_under( mLua, "Table", cLua_RefObject );
|
1610
|
+
rb_define_method( cLua_Table, "to_array", RUBY_METHOD_FUNC(rlua_Table_to_array), 0 );
|
1611
|
+
rb_define_method( cLua_Table, "to_hash", RUBY_METHOD_FUNC(rlua_Table_to_hash), 0 );
|
1612
|
+
rb_define_method( cLua_Table, "each_ipair", RUBY_METHOD_FUNC(rlua_Table_each_ipair), 0 );
|
1613
|
+
rb_define_method( cLua_Table, "each_ikey", RUBY_METHOD_FUNC(rlua_Table_each_ikey), 0 );
|
1614
|
+
rb_define_method( cLua_Table, "each_ivalue", RUBY_METHOD_FUNC(rlua_Table_each_ivalue), 0 );
|
1615
|
+
rb_define_method( cLua_Table, "each_pair", RUBY_METHOD_FUNC(rlua_Table_each_pair), 0 );
|
1616
|
+
rb_define_method( cLua_Table, "each_key", RUBY_METHOD_FUNC(rlua_Table_each_key), 0 );
|
1617
|
+
rb_define_method( cLua_Table, "each_value", RUBY_METHOD_FUNC(rlua_Table_each_value), 0 );
|
1618
|
+
rb_define_alias( cLua_Table, "each", "each_pair" );
|
1619
|
+
rb_define_alias( cLua_Table, "each_index", "each_ikey" );
|
1620
|
+
}
|
1621
|
+
|
1622
|
+
} // extern "C"
|
1623
|
+
|