Module: DL::Importable::Internal
- Included in:
- DL::Importable
- Defined in:
- lib/dl/import.rb,
lib/dl/struct.rb
Defined Under Namespace
Classes: Memory, Struct, Union
Instance Method Summary collapse
- #[](name) ⇒ Object
- #_args_ ⇒ Object
- #_retval_ ⇒ Object
-
#callback(proto) ⇒ Object
example: callback “int method_name(int, char*)”.
- #define_struct(contents) ⇒ Object (also: #struct)
- #define_union(contents) ⇒ Object (also: #union)
- #dlload(*libnames) ⇒ Object (also: #dllink)
- #encode_argument_types(tys) ⇒ Object
-
#extern(proto) ⇒ Object
example: extern “int strlen(char*)”.
-
#import(name, rettype, argtypes = nil) ⇒ Object
example: import(“get_length”, “int”, [“void*”, “int”]).
- #init_sym ⇒ Object
- #init_types ⇒ Object
- #parse_cproto(proto) ⇒ Object
-
#symbol(name, ty = nil) ⇒ Object
example: symbol “foo_value” symbol “foo_func”, “IIP”.
-
#typealias(alias_type, ty1, enc1 = nil, dec1 = nil, ty2 = nil, enc2 = nil, dec2 = nil) ⇒ Object
example: typealias(“uint”, “unsigned int”).
Instance Method Details
#[](name) ⇒ Object
19 20 21 |
# File 'lib/dl/import.rb', line 19 def [](name) return @SYM[name.to_s][0] end |
#_args_ ⇒ Object
180 181 182 |
# File 'lib/dl/import.rb', line 180 def _args_ return @args end |
#_retval_ ⇒ Object
184 185 186 |
# File 'lib/dl/import.rb', line 184 def _retval_ return @retval end |
#callback(proto) ⇒ Object
example:
callback "int method_name(int, char*)"
67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 |
# File 'lib/dl/import.rb', line 67 def callback(proto) func,ret,args = parse_cproto(proto) init_types() init_sym() rty,renc,rdec = @types.encode_return_type(ret) if( !rty ) raise(TypeError, "unsupported type: #{ret}") end ty,enc,dec = encode_argument_types(args) symty = rty + ty module_eval("module_function :#{func}") sym = module_eval([ "DL::callback(\"#{symty}\"){|*args|", " sym,rdec,enc,dec = @SYM['#{func}']", " args = enc.call(args) if enc", " r,rs = #{func}(*args)", " r = renc.call(r) if rdec", " rs = dec.call(rs) if (dec && rs)", " @retval = r", " @args = rs", " r", "}", ].join("\n")) @SYM[func] = [sym,rdec,enc,dec] return sym end |
#define_struct(contents) ⇒ Object Also known as: struct
9 10 11 12 |
# File 'lib/dl/struct.rb', line 9 def define_struct(contents) init_types() Struct.new(@types, contents) end |
#define_union(contents) ⇒ Object Also known as: union
15 16 17 18 |
# File 'lib/dl/struct.rb', line 15 def define_union(contents) init_types() Union.new(@types, contents) end |
#dlload(*libnames) ⇒ Object Also known as: dllink
23 24 25 26 27 28 29 30 31 32 33 |
# File 'lib/dl/import.rb', line 23 def dlload(*libnames) if( !defined?(@LIBS) ) @LIBS = [] end libnames.each{|libname| if( !LIB_MAP[libname] ) LIB_MAP[libname] = DL.dlopen(libname) end @LIBS.push(LIB_MAP[libname]) } end |
#encode_argument_types(tys) ⇒ Object
188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 |
# File 'lib/dl/import.rb', line 188 def encode_argument_types(tys) init_types() encty = [] enc = nil dec = nil tys.each_with_index{|ty,idx| ty,c1,c2 = @types.encode_argument_type(ty) if( !ty ) raise(TypeError, "unsupported type: #{ty}") end encty.push(ty) if( enc ) if( c1 ) conv1 = enc enc = proc{|v| v = conv1.call(v); v[idx] = c1.call(v[idx]); v} end else if( c1 ) enc = proc{|v| v[idx] = c1.call(v[idx]); v} end end if( dec ) if( c2 ) conv2 = dec dec = proc{|v| v = conv2.call(v); v[idx] = c2.call(v[idx]); v} end else if( c2 ) dec = proc{|v| v[idx] = c2.call(v[idx]); v} end end } return [encty.join, enc, dec] end |
#extern(proto) ⇒ Object
example:
extern "int strlen(char*)"
59 60 61 62 |
# File 'lib/dl/import.rb', line 59 def extern(proto) func,ret,args = parse_cproto(proto) return import(func, ret, args) end |
#import(name, rettype, argtypes = nil) ⇒ Object
example:
import("get_length", "int", ["void*", "int"])
134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 |
# File 'lib/dl/import.rb', line 134 def import(name, rettype, argtypes = nil) init_types() init_sym() rty,_,rdec = @types.encode_return_type(rettype) if( !rty ) raise(TypeError, "unsupported type: #{rettype}") end ty,enc,dec = encode_argument_types(argtypes) symty = rty + ty sym = symbol(name, symty) mname = name.dup if( ?A <= mname[0] && mname[0] <= ?Z ) mname[0,1] = mname[0,1].downcase end @SYM[mname] = [sym,rdec,enc,dec] module_eval [ "def #{mname}(*args)", " sym,rdec,enc,dec = @SYM['#{mname}']", " args = enc.call(args) if enc", if( $DEBUG ) " p \"[DL] call #{mname} with \#{args.inspect}\"" else "" end, " r,rs = sym.call(*args)", if( $DEBUG ) " p \"[DL] retval=\#{r.inspect} args=\#{rs.inspect}\"" else "" end, " r = rdec.call(r) if rdec", " rs = dec.call(rs) if dec", " @retval = r", " @args = rs", " return r", "end", "module_function :#{mname}", ].join("\n") return sym end |
#init_sym ⇒ Object
15 16 17 |
# File 'lib/dl/import.rb', line 15 def init_sym() @SYM ||= {} end |
#init_types ⇒ Object
11 12 13 |
# File 'lib/dl/import.rb', line 11 def init_types() @types ||= ::DL::Types.new end |
#parse_cproto(proto) ⇒ Object
36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 |
# File 'lib/dl/import.rb', line 36 def parse_cproto(proto) proto = proto.gsub(/\s+/, " ").strip case proto when /^([\d\w\*_\s]+)\(([\d\w\*_\s\,\[\]]*)\)$/ ret = $1 args = $2.strip() ret = ret.split(/\s+/) args = args.split(/\s*,\s*/) func = ret.pop() if( func =~ /^\*/ ) func.gsub!(/^\*+/,"") ret.push("*") end ret = ret.join(" ") return [func, ret, args] else raise(RuntimeError,"can't parse the function prototype: #{proto}") end end |
#symbol(name, ty = nil) ⇒ Object
example:
symbol "foo_value"
symbol "foo_func", "IIP"
112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 |
# File 'lib/dl/import.rb', line 112 def symbol(name, ty = nil) sym = nil @LIBS.each{|lib| begin if( ty ) sym = lib[name, ty] else sym = lib[name] end rescue next end } if( !sym ) raise(RuntimeError, "can't find the symbol `#{name}'") end return sym end |
#typealias(alias_type, ty1, enc1 = nil, dec1 = nil, ty2 = nil, enc2 = nil, dec2 = nil) ⇒ Object
example:
typealias("uint", "unsigned int")
102 103 104 105 106 |
# File 'lib/dl/import.rb', line 102 def typealias(alias_type, ty1, enc1=nil, dec1=nil, ty2=nil, enc2=nil, dec2=nil) init_types() @types.typealias(alias_type, ty1, enc1, dec1, ty2||ty1, enc2, dec2) end |