/* Serene --- Yet an other Lisp Copyright (c) 2020 Sameer Rahmani This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 2 of the License. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ package core import ( "errors" "serene-lang.org/bootstrap/pkg/ast" ) // Def defines a global binding in the current namespace. The first // arguments in `args` has to be a symbol ( none ns qualified ) and // the second param should be the value of the binding func Def(rt *Runtime, scope IScope, args *List) (IExpr, error) { // TODO: Add support for docstrings and meta switch args.Count() { case 2: name := args.First() if name.GetType() != ast.Symbol { return nil, errors.New("the first argument of 'def' has to be a symbol") } sym := name.(*Symbol) valueExpr := args.Rest().First() value, err := EvalForms(rt, scope, valueExpr) if err != nil { return nil, err } ns := rt.CurrentNS() ns.DefineGlobal(sym.GetName(), value, true) return sym, nil } return nil, errors.New("'def' form need at least 2 arguments") } // Fn defines a function inside the given scope `scope` with the given `args`. // `args` contains the arugment list, docstring and body of the function. func Fn(rt *Runtime, scope IScope, args *List) (IExpr, error) { if args.Count() < 1 { return nil, errors.New("'fn' needs at least an arguments list") } var params IColl body := MakeEmptyBlock() arguments := args.First() // TODO: Add vector in here // Or any other icoll if arguments.GetType() == ast.List { params = arguments.(IColl) } if args.Count() > 1 { body.SetContent(args.Rest().(*List).ToSlice()) } return MakeFunction(scope, params, body), nil }