Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/trunk/src/external/tolua/lua/function.lua @ 8298

Last change on this file since 8298 was 5738, checked in by landauf, 16 years ago

merged libraries2 back to trunk

  • Property svn:eol-style set to native
File size: 21.4 KB
RevLine 
[2087]1-- tolua: function class
2-- Written by Waldemar Celes
3-- TeCGraf/PUC-Rio
4-- Jul 1998
5-- $Id: function.lua 1597 2008-03-11 22:25:01Z ice-drezday $
6
7-- This code is free software; you can redistribute it and/or modify it.
8-- The software provided hereunder is on an "as is" basis, and
9-- the author has no obligation to provide maintenance, support, updates,
10-- enhancements, or modifications.
11
12
13-- CEGUILua mod
14-- exception handling
15-- patch from Tov
16-- modded by Lindquist
17exceptionDefs = exceptionDefs or {}
18exceptionDefs["std::exception"]             = {}
19exceptionDefs["std::exception"]["var"]      = "&e"
20exceptionDefs["std::exception"]["c_str"]    = "e.what()"
21
22exceptionDefs["any"]                        = {}
23exceptionDefs["any"]["var"]                 = ""
24exceptionDefs["any"]["c_str"]               = '"Unknown"'
25
26exceptionMessageBufferSize = 512
27
28function outputExceptionError(f,e,errBuf)
29    -- if the exception is not "..." then use the "c_str" info the get a real exception message
30    local messageC_str = true
31    if e.name == "any" then
32        messageC_str = false
33    end
34
35    -- make a default e.ret if empty
36    if not e.ret or e.ret == "" then
37        e.ret = "nil,message"
38    end
39
40    -- create a default exceptionDef if we dont have one
41    if not exceptionDefs[e.name] then
42        exceptionDefs[e.name] = {}
43        exceptionDefs[e.name].var = "&e"
44        exceptionDefs[e.name].c_str = '"Unknown"'
45    end
46
47    -- print catch header
48    local nameToEcho = e.name
49    if nameToEcho == "any" then
50        nameToEcho = "..."
51    end
52    if e.ret == "nil" then
53        output("catch(",nameToEcho," CEGUIDeadException(",exceptionDefs[e.name].var,"))\n{\n")
[2710]54    else
55        output("catch(",nameToEcho,exceptionDefs[e.name].var,")\n{\n")
56    end
[2087]57
58    -- if just a nil
59    if e.ret == "nil" then
60        output("return 0;\n")
61    -- if error should be raised
62    elseif string.find(e.ret,"error") then
63        if messageC_str then
64            output("snprintf(errorBuffer,"..exceptionMessageBufferSize..",\"Exception of type '"..e.name.."' was thrown by function '"..f.."'\\nMessage: %s\","..exceptionDefs[e.name].c_str..");\n")
65        else
66            output("snprintf(errorBuffer,"..exceptionMessageBufferSize..",\"Unknown exception thrown by function '"..f.."'\");\n")
67        end
68        output("errorDoIt = true;\n")
69    -- else go through the returns
70    else
71        -- buffer for message
72        if string.find(e.ret,"message") and messageC_str and errBuf == false then
73            output("char errorBuffer["..exceptionMessageBufferSize.."];\n")
74        end
75        local numrets = 0
76        local retpat = "(%w+),?"
77        local i,j,retval = string.find(e.ret,retpat)
78        while i do
79            local code = ""
80
81            -- NIL
82            if retval == "nil" then
83                code = "tolua_pushnil(tolua_S);\n"
84
85            -- MESSAGE
86            elseif retval == "message" then
87                if messageC_str then
88                    code = "snprintf(errorBuffer,"..exceptionMessageBufferSize..",\"Exception of type '"..e.name.."' was thrown by function '"..f.."'\\nMessage: %s\","..exceptionDefs[e.name].c_str..");\ntolua_pushstring(tolua_S,errorBuffer);\n"
89                else
90                    code = "tolua_pushstring(tolua_S,\"Unknown exception thrown by function '"..f.."'\");\n"
91                end
92
93            -- TRUE
94            elseif retval == "true" then
95                code = "tolua_pushboolean(tolua_S, 1);\n"
96
97            -- FALSE
98            elseif retval == "false" then
99                code = "tolua_pushboolean(tolua_S, 0);\n"
100            end
101
102            -- print code for this return value
103            if code ~= "" then
104                output(code)
105                numrets = numrets + 1
106            end
107
108            -- next return value
109            i,j,retval = string.find(e.ret,retpat,j+1)
110        end
111        output("return ",numrets,";\n")
112    end
113
114    -- print catch footer
115    output("}\n")
116end
117
118function outputExceptionCatchBlocks(func,throws,err)
119    for i=1,table.getn(throws) do
120        outputExceptionError(func, throws[i], err)
121    end
122
123    -- if an error should be raised, we do it here
124    if err then
125        output("if (errorDoIt) {\n")
126        output("luaL_error(tolua_S,errorBuffer);\n")
127        output("}\n")
128    end
129end
130
131
132-- Function class
133-- Represents a function or a class method.
134-- The following fields are stored:
135--  mod  = type modifiers
136--  type = type
137--  ptr  = "*" or "&", if representing a pointer or a reference
138--  name = name
139--  lname = lua name
140--  args  = list of argument declarations
141--  const = if it is a method receiving a const "this".
142classFunction = {
[2710]143    mod = '',
144    type = '',
145    ptr = '',
146    name = '',
147    args = {n=0},
148    const = '',
[2087]149}
150classFunction.__index = classFunction
151setmetatable(classFunction,classFeature)
152
153-- declare tags
154function classFunction:decltype ()
[2710]155    self.type = typevar(self.type)
156    if strfind(self.mod,'const') then
157        self.type = 'const '..self.type
158        self.mod = gsub(self.mod,'const','')
159    end
160    local i=1
161    while self.args[i] do
162        self.args[i]:decltype()
163        i = i+1
164    end
[2087]165end
166
167
168-- Write binding function
169-- Outputs C/C++ binding function.
170function classFunction:supcode (local_constructor)
171
[2710]172    local overload = strsub(self.cname,-2,-1) - 1  -- indicate overloaded func
173    local nret = 0      -- number of returned values
174    local class = self:inclass()
175    local _,_,static = strfind(self.mod,'^%s*(static)')
176    if class then
[2087]177
[2710]178        if self.name == 'new' and self.parent.flags.pure_virtual then
179            -- no constructor for classes with pure virtual methods
180            return
181        end
[2087]182
[2710]183        if local_constructor then
184           output("/* method: new_local of class ",class," */")
185       else
186           output("/* method:",self.name," of class ",class," */")
187       end
188    else
189        output("/* function:",self.name," */")
190    end
[2087]191
[2710]192    if local_constructor then
193        output("#ifndef TOLUA_DISABLE_"..self.cname.."_local")
194        output("\nstatic int",self.cname.."_local","(lua_State* tolua_S)")
195    else
196        output("#ifndef TOLUA_DISABLE_"..self.cname)
197        output("\nstatic int",self.cname,"(lua_State* tolua_S)")
198    end
199    output("{")
[2087]200
[2710]201    -- check types
202    if overload < 0 then
203        output('#ifndef TOLUA_RELEASE\n')
204    end
205    output(' tolua_Error tolua_err;')
206    output(' if (\n')
207    -- check self
208    local narg
209    if class then narg=2 else narg=1 end
210    if class then
211        local func = 'tolua_isusertype'
212        local type = self.parent.type
213        if self.name=='new' or static~=nil then
214            func = 'tolua_isusertable'
215            type = self.parent.type
216        end
217        if self.const ~= '' then
218            type = "const "..type
219        end
220        output('     !'..func..'(tolua_S,1,"'..type..'",0,&tolua_err) ||\n')
221    end
222    -- check args
223    if self.args[1].type ~= 'void' then
224        local i=1
225        while self.args[i] do
226            local btype = isbasic(self.args[i].type)
227            if btype ~= 'value' and btype ~= 'state' then
228                output('     !'..self.args[i]:outchecktype(narg)..' ||\n')
229            end
230            if btype ~= 'state' then
231                narg = narg+1
232            end
233            i = i+1
234        end
235    end
236    -- check end of list
237    output('     !tolua_isnoobj(tolua_S,'..narg..',&tolua_err)\n )')
238    output('  goto tolua_lerror;')
[2087]239
[2710]240    output(' else\n')
241    if overload < 0 then
242        output('#endif\n')
243    end
244    output(' {')
[2087]245
[2710]246    -- declare self, if the case
247    local narg
248    if class then narg=2 else narg=1 end
249    if class and self.name~='new' and static==nil then
250        output(' ',self.const,self.parent.type,'*','self = ')
251        output('(',self.const,self.parent.type,'*) ')
252        output('tolua_tousertype(tolua_S,1,0);')
253    elseif static then
254        _,_,self.mod = strfind(self.mod,'^%s*static%s%s*(.*)')
255    end
256    -- declare parameters
257    if self.args[1].type ~= 'void' then
258        local i=1
259        while self.args[i] do
260            self.args[i]:declare(narg)
261            if isbasic(self.args[i].type) ~= "state" then
262                narg = narg+1
263            end
264            i = i+1
265        end
266    end
[2087]267
[2710]268    -- check self
269    if class and self.name~='new' and static==nil then
270        output('#ifndef TOLUA_RELEASE\n')
271        output('  if (!self) tolua_error(tolua_S,"invalid \'self\' in function \''..self.name..'\'",NULL);');
272        output('#endif\n')
273    end
[2087]274
[2710]275    -- get array element values
276    if class then narg=2 else narg=1 end
277    if self.args[1].type ~= 'void' then
278        local i=1
279        while self.args[i] do
280            self.args[i]:getarray(narg)
281            narg = narg+1
282            i = i+1
283        end
284    end
[2087]285
[2710]286    --------------------------------------------------
287    -- CEGUILua mod
288    -- init exception handling
289    local throws = false
290    do
291        local pattern = "tolua_throws|.*|"
292        local i,j = string.find(self.mod, pattern)
293        if i then
294            throws = {}
295            -- ensure table is empty.  Used to be: table.setn(throws,0)
296            for x in pairs(throws) do
297                throws[x] = nil
298            end
299            local excepts = string.sub(self.mod, i+12,j)
300            local epattern = "|.-|"
301            local i,j = string.find(excepts, epattern)
302            while i do
303                local e = string.sub(excepts,i+1,j-1)
304                local _,_,name,rest = string.find(e, "([%w:_]+),?(.*)")
305                table.insert(throws,{name=name, ret=rest})
306                i,j = string.find(excepts, epattern, j)
307            end
308            self.mod = string.gsub(self.mod, pattern, "")
309        end
310    end
311    local exRaiseError = false
312    --------------------------------------------------
[2087]313
[2710]314    local out = string.find(self.mod, "tolua_outside")
[2087]315
[2710]316    ---------------
317    -- CEGUILua mod
318    -- remove "tolua_outside" from self.mod
319    if out then
320        self.mod = string.gsub(self.mod, "tolua_outside", "")
321    end
[2087]322
[2710]323    -- call function
324    if class and self.name=='delete' then
325        output('  delete self;')
326    elseif class and self.name == 'operator&[]' then
327        if flags['1'] then -- for compatibility with tolua5 ?
328            output('  self->operator[](',self.args[1].name,'-1) = ',self.args[2].name,';')
329        else
330            output('  self->operator[](',self.args[1].name,') = ',self.args[2].name,';')
[2087]331        end
[2710]332    else
333    -- CEGUILua mod begin- throws
334    if throws then
335        for i=1,table.getn(throws) do
336            if string.find(throws[i].ret, "error") then
337                output("char errorBuffer["..exceptionMessageBufferSize.."];\n")
338                output("bool errorDoIt = false;\n")
339                exRaiseError = true
340                break
341            end
342        end
343        output("try\n")
[2087]344    end
[2710]345    -- CEGUILua mod end - throws
346    output('  {')
347    if self.type ~= '' and self.type ~= 'void' then
348        output('  ',self.mod,self.type,self.ptr,'tolua_ret = ')
349        output('(',self.mod,self.type,self.ptr,') ')
350    else
351        output('  ')
352    end
353    if class and self.name=='new' then
354        output('new',self.type,'(')
355    elseif class and static then
356        if out then
357            output(self.name,'(')
358        else
359            output(class..'::'..self.name,'(')
360        end
361    elseif class then
362        if out then
363            output(self.name,'(')
364        else
365            if self.cast_operator then
366                output('static_cast<',self.mod,self.type,self.ptr,'>(*self')
367            else
368                output('self->'..self.name,'(')
369            end
370        end
371    else
372        output(self.name,'(')
373    end
[2087]374
[2710]375    if out and not static then
376        output('self')
377        if self.args[1] and self.args[1].name ~= '' then
378            output(',')
379        end
380    end
381    -- write parameters
382    local i=1
383    while self.args[i] do
384        self.args[i]:passpar()
385        i = i+1
386        if self.args[i] then
387            output(',')
388        end
389    end
[2087]390
[2710]391    if class and self.name == 'operator[]' and flags['1'] then
392        output('-1);')
[2087]393    else
[2710]394        output(');')
[2087]395    end
396
[2710]397    -- return values
398    if self.type ~= '' and self.type ~= 'void' then
399        nret = nret + 1
400        local t,ct = isbasic(self.type)
401        if t then
402            if self.cast_operator and _basic_raw_push[t] then
403                output('   ',_basic_raw_push[t],'(tolua_S,(',ct,')tolua_ret);')
404            else
405                output('   tolua_push'..t..'(tolua_S,(',ct,')tolua_ret);')
406            end
407            else
408                t = self.type
409                new_t = string.gsub(t, "const%s+", "")
410                if self.ptr == '' then
411                    output('   {')
412                    output('#ifdef __cplusplus\n')
413                    output('    void* tolua_obj = new',new_t,'(tolua_ret);')
414                    output('    tolua_pushusertype_and_takeownership(tolua_S,tolua_obj,"',t,'");')
415                    output('#else\n')
416                    output('    void* tolua_obj = tolua_copy(tolua_S,(void*)&tolua_ret,sizeof(',t,'));')
417                    output('    tolua_pushusertype_and_takeownership(tolua_S,tolua_obj,"',t,'");')
418                    output('#endif\n')
419                    output('   }')
420                elseif self.ptr == '&' then
421                    output('   tolua_pushusertype(tolua_S,(void*)&tolua_ret,"',t,'");')
422                else
423                    if local_constructor then
424                        output('   tolua_pushusertype_and_takeownership(tolua_S,(void *)tolua_ret,"',t,'");')
425                    else
426                        output('   tolua_pushusertype(tolua_S,(void*)tolua_ret,"',t,'");')
427                    end
428                end
429            end
430        end
431        local i=1
432        while self.args[i] do
433            nret = nret + self.args[i]:retvalue()
434            i = i+1
435        end
436        output('  }')
[2087]437
[2710]438        ------------------------------------------
439        -- CEGUILua mod
440        -- finish exception handling
441        -- catch
442        if throws then
443            outputExceptionCatchBlocks(self.name, throws, exRaiseError)
444        end
445        ------------------------------------------
[2087]446
[2710]447        -- set array element values
448        if class then narg=2 else narg=1 end
449        if self.args[1].type ~= 'void' then
450            local i=1
451            while self.args[i] do
452                self.args[i]:setarray(narg)
453                narg = narg+1
454                i = i+1
455            end
456        end
[2087]457
[2710]458        -- free dynamically allocated array
459        if self.args[1].type ~= 'void' then
460            local i=1
461            while self.args[i] do
462                self.args[i]:freearray()
463                i = i+1
464            end
465        end
466    end
[2087]467
[2710]468    output(' }')
469    output(' return '..nret..';')
[2087]470
[2710]471    -- call overloaded function or generate error
472    if overload < 0 then
[2087]473
[2710]474        output('#ifndef TOLUA_RELEASE\n')
475        output('tolua_lerror:\n')
476        output(' tolua_error(tolua_S,"#ferror in function \''..self.lname..'\'.",&tolua_err);')
477        output(' return 0;')
478        output('#endif\n')
479    else
480        local _local = ""
481        if local_constructor then
482            _local = "_local"
483        end
484        output('tolua_lerror:\n')
485        output(' return '..strsub(self.cname,1,-3)..format("%02d",overload).._local..'(tolua_S);')
486    end
487    output('}')
488    output('#endif //#ifndef TOLUA_DISABLE\n')
489    output('\n')
[2087]490
[2710]491    -- recursive call to write local constructor
492    if class and self.name=='new' and not local_constructor then
[2087]493
[2710]494        self:supcode(1)
495    end
496
[2087]497end
498
499
500-- register function
501function classFunction:register (pre)
502
[2710]503    if not self:check_public_access() then
504        return
505    end
[2087]506
[2710]507    if self.name == 'new' and self.parent.flags.pure_virtual then
508        -- no constructor for classes with pure virtual methods
509        return
510    end
[2087]511
[2710]512    output(pre..'tolua_function(tolua_S,"'..self.lname..'",'..self.cname..');')
513    if self.name == 'new' then
514        output(pre..'tolua_function(tolua_S,"new_local",'..self.cname..'_local);')
515        output(pre..'tolua_function(tolua_S,".call",'..self.cname..'_local);')
516        --output(' tolua_set_call_event(tolua_S,'..self.cname..'_local, "'..self.parent.type..'");')
517    end
[2087]518end
519
520-- Print method
521function classFunction:print (ident,close)
[2710]522    print(ident.."Function{")
523    print(ident.." mod  = '"..self.mod.."',")
524    print(ident.." type = '"..self.type.."',")
525    print(ident.." ptr  = '"..self.ptr.."',")
526    print(ident.." name = '"..self.name.."',")
527    print(ident.." lname = '"..self.lname.."',")
528    print(ident.." const = '"..self.const.."',")
529    print(ident.." cname = '"..self.cname.."',")
530    print(ident.." lname = '"..self.lname.."',")
531    print(ident.." args = {")
532    local i=1
533    while self.args[i] do
534        self.args[i]:print(ident.."  ",",")
535        i = i+1
536    end
537    print(ident.." }")
538    print(ident.."}"..close)
[2087]539end
540
541-- check if it returns an object by value
542function classFunction:requirecollection (t)
[2710]543    local r = false
544    if self.type ~= '' and not isbasic(self.type) and self.ptr=='' then
545        local type = gsub(self.type,"%s*const%s+","")
546        t[type] = "tolua_collect_" .. clean_template(type)
547        r = true
548    end
549    local i=1
550    while self.args[i] do
551        r = self.args[i]:requirecollection(t) or r
552        i = i+1
553    end
554    return r
[2087]555end
556
557-- determine lua function name overload
558function classFunction:overload ()
[2710]559    return self.parent:overload(self.lname)
[2087]560end
561
562
563function param_object(par) -- returns true if the parameter has an object as its default value
564
[2710]565    if not string.find(par, '=') then return false end -- it has no default value
[2087]566
[2710]567    local _,_,def = string.find(par, "=(.*)$")
[2087]568
[2710]569    if string.find(par, "|") then -- a list of flags
[2087]570
[2710]571        return true
572    end
[2087]573
[2710]574    if string.find(par, "%*") then -- it's a pointer with a default value
[2087]575
[2710]576        if string.find(par, '=%s*new') then -- it's a pointer with an instance as default parameter.. is that valid?
577            return true
578        end
579        return false -- default value is 'NULL' or something
580    end
[2087]581
582
[2710]583    if string.find(par, "[%(&]") then
584        return true
585    end -- default value is a constructor call (most likely for a const reference)
[2087]586
[2710]587    --if string.find(par, "&") then
[2087]588
[2710]589    --    if string.find(def, ":") or string.find(def, "^%s*new%s+") then
[2087]590
[2710]591    --        -- it's a reference with default to something like Class::member, or 'new Class'
592    --        return true
593    --    end
594    --end
[2087]595
[2710]596    return false -- ?
[2087]597end
598
599function strip_last_arg(all_args, last_arg) -- strips the default value from the last argument
600
[2710]601    local _,_,s_arg = string.find(last_arg, "^([^=]+)")
602    last_arg = string.gsub(last_arg, "([%%%(%)])", "%%%1");
603    all_args = string.gsub(all_args, "%s*,%s*"..last_arg.."%s*%)%s*$", ")")
604    return all_args, s_arg
[2087]605end
606
607
608
609-- Internal constructor
610function _Function (t)
[2710]611    setmetatable(t,classFunction)
[2087]612
[2710]613    if t.const ~= 'const' and t.const ~= '' then
614        error("#invalid 'const' specification")
615    end
[2087]616
[2710]617    append(t)
618    if t:inclass() then
619        --print ('t.name is '..t.name..', parent.name is '..t.parent.name)
620        if string.gsub(t.name, "%b<>", "") == string.gsub(t.parent.original_name or t.parent.name, "%b<>", "") then
621            t.name = 'new'
622            t.lname = 'new'
623            t.parent._new = true
624            t.type = t.parent.name
625            t.ptr = '*'
626        elseif string.gsub(t.name, "%b<>", "") == '~'..string.gsub(t.parent.original_name or t.parent.name, "%b<>", "") then
627            t.name = 'delete'
628            t.lname = 'delete'
629            t.parent._delete = true
630        end
631    end
632    t.cname = t:cfuncname("tolua")..t:overload(t)
633    return t
[2087]634end
635
636-- Constructor
637-- Expects three strings: one representing the function declaration,
638-- another representing the argument list, and the third representing
639-- the "const" or empty string.
640function Function (d,a,c)
[2710]641    --local t = split(strsub(a,2,-2),',') -- eliminate braces
642    --local t = split_params(strsub(a,2,-2))
[2087]643
[2710]644    if not flags['W'] and string.find(a, "%.%.%.%s*%)") then
[2087]645
[2710]646        warning("Functions with variable arguments (`...') are not supported. Ignoring "..d..a..c)
647        return nil
648    end
[2087]649
650
[2710]651    local i=1
652    local l = {n=0}
[2087]653
[2710]654    a = string.gsub(a, "%s*([%(%)])%s*", "%1")
655    local t,strip,last = strip_pars(strsub(a,2,-2));
656    if strip then
657        --local ns = string.sub(strsub(a,1,-2), 1, -(string.len(last)+1))
658        local ns = join(t, ",", 1, last-1)
[2087]659
[2710]660        ns = "("..string.gsub(ns, "%s*,%s*$", "")..')'
661        --ns = strip_defaults(ns)
[2087]662
[2710]663        Function(d, ns, c)
664        for i=1,last do
665            t[i] = string.gsub(t[i], "=.*$", "")
666        end
667    end
[2087]668
[2710]669    while t[i] do
670        l.n = l.n+1
671        l[l.n] = Declaration(t[i],'var',true)
672        i = i+1
673    end
674    local f = Declaration(d,'func')
675    f.args = l
676    f.const = c
677    return _Function(f)
[2087]678end
679
680function join(t, sep, first, last)
681
[2710]682    first = first or 1
683    last = last or table.getn(t)
684    local lsep = ""
685    local ret = ""
686    local loop = false
687    for i = first,last do
[2087]688
[2710]689        ret = ret..lsep..t[i]
690        lsep = sep
691        loop = true
692    end
693    if not loop then
694        return ""
695    end
[2087]696
[2710]697    return ret
[2087]698end
699
700function strip_pars(s)
701
[2710]702    local t = split_c_tokens(s, ',')
703    local strip = false
704    local last
[2087]705
[2710]706    for i=t.n,1,-1 do
[2087]707
[2710]708        if not strip and param_object(t[i]) then
709            last = i
710            strip = true
711        end
712        --if strip then
713        --    t[i] = string.gsub(t[i], "=.*$", "")
714        --end
715    end
[2087]716
[2710]717    return t,strip,last
[2087]718
719end
720
721function strip_defaults(s)
722
[2710]723    s = string.gsub(s, "^%(", "")
724    s = string.gsub(s, "%)$", "")
[2087]725
[2710]726    local t = split_c_tokens(s, ",")
727    local sep, ret = "",""
728    for i=1,t.n do
729        t[i] = string.gsub(t[i], "=.*$", "")
730        ret = ret..sep..t[i]
731        sep = ","
732    end
[2087]733
[2710]734    return "("..ret..")"
[2087]735end
736
737
Note: See TracBrowser for help on using the repository browser.