1. Introduction

daScript is high-performance statically strong typed scripting language, designed to be high-performance as embeddable “scripting” language for real-time applications (like games).

daScript offers a wide range of features like strong static typing, generic programming with iterative type inference, Ruby-like blocks, semantic indenting, native machine types, ahead-of-time “compilation” to C++ and fast and simplified bindings to C++ program.

It’s philsophy is build around modified Zen of Python.

  • Performance counts.

  • But not at the cost of safety.

  • Unless is explicitly unsafe to be performant.

  • Readability counts.

  • Explicit is better than implicit.

  • Simple is better than complex.

  • Complex is better than complicated.

  • Flat is better than nested.

daScript is supposed to work as “host data processor”. While it is technically possible to mantain persistent state within script context (with a certain option set), daScript is designed to transform your host (C++) data/implement scripted behaviours.

In a certain sense, it is pure functional - i.e. all persistent state is out of scope of scripting context, and script’s state is temporal by it’s nature. So, the memory model and managment of persistent state becoming are or responsibility of application, and leads to extremely simple and fast memory model in the daScript itself.

1.1. Performance.

In a real world scenarios, it’s interpretation is 10+ times faster than LuaJIT without JIT (and can be even faster than LuaJIT with JIT). Which is probably even more important for embedded scripting languages, it’s (both-ways) interop with C++ is extremely fast, order of magnitude faster than such in most popular scripting languages. Fast “call from C++ to daScript” allows to use daScript for simple stored procedures, and makes it’s ECS/Data Oriented Design friendly language. Fast call C++ from daScript allows to write performant scripts which are processing host (C++) data, or relies on bound host (C++) functions.

It also allows Ahead-of-Time compilation, which is not only possible on all platforms (unlike JIT), but also always faster/not-slower (JIT is known to sometimes slow down scripts).

daScript already has implemented AoT (C++ transpiler) which produces code more or less similar with C++11 performance.

Table with performance comparisons on a syntetic samples.

1.2. How it looks?

Mandatory fibonacci samples

def fibR(n)
   if (n < 2)
       return n
   else
       return fibR(n - 1) + fibR(n - 2)

def fibI(n)
   var last = 0
   var cur = 1
   for i in range(0, n - 1)
       let tmp = cur
       cur += last
       last = tmp
   return cur

1.3. Generic programming and type system

Although above sample seem to be dynamically typed, it is actually a generic programming. The actual instance of fibI/fibR function is strong typed and basically is just accepting and returning int. This is similar to templates in C++ (although C++ is not atrong-typed language) or ML. Generic programming in daScript allows very powerful compile-time type reflection mechanisms, significally simplifying writing optimal and clear code. Unlike C++ with it’s SFINAE, you can use comon conditions (if) in order to change instance of function depending on type info of arguments. Consider the following example:

def setSomeField(var obj; val)
    if typeinfo(has_field<someField> obj)
          obj.someField = val

this function will set someField in provided argument if it is a struct with someField member.

For more info see Generic programming).

1.4. Features

It’s (not)full list of features includes:

  • strong typing

  • Ruby-like blocks

  • tables

  • arrays

  • string-builder

  • native (C++ friendly) interop

  • generics

  • semantic indenting

  • ECS-friendly interop

  • easy-to-extend type system

  • etc.