nac3-spec/README.md
2020-12-15 13:09:32 +08:00

3.4 KiB

NAC3 Specification

Specification and discussions about language design.

Referencing Python Variables

Not decided yet, whether require function annotation or only allow reference to variables.

The kernel is allowed to read Python variables.

Unbounded identifiers would be considered as Python variables, no object is allowed. The value would be evaluated at compile time, subsequent modification in the host would not be known by the kernel. Basically, only allow lookup, no evaluation.

(Bad) Alternative: Evaluate the unbounded identifier, allowing functions and objects. It would easier to write the program as we don't have to manually remove function calls in the kernel and store the result in a global variable. However, this would potentially cause confusion, as the possibly side-effectful function would be evaluated only once during compilation.

(Better) Alternative: only evaluate functions marked as pure? Not sure if we can access custom function annotation. We have to rely on the user to uphold the guarantee.

Example for a potentially confusing case:

from artiq.experiment import *
counter = 0
def get_id():
    counter += 1
    return counter

class Foo(EnvExperiment):
    @kernel
    def run(self):
        param = get_id()
        # do something...
        result = param
        return result

Example for a totally valid case:

from artiq.experiment import *
def get_param(x):
    return x**2

class Foo(EnvExperiment):
    @kernel
    def run(self):
        # unbounded function call disallowed
        param = get_param(123)
        # do something...
        result = param
        return result

This would not be allowed, and must be translated into this

from artiq.experiment import *
def get_param(x):
    return x**2

param_123 = get_param(123)
class Foo(EnvExperiment):
    @kernel
    def run(self):
        param = param_123
        # do something...
        result = param
        return result

Type

Decided

  • Parametric polymorphism: use Python type variable.
  • Normal functions: require full type signature.
  • RPC: optional parameter type signature, require return type signature.
  • No implicit coercion

Undecided

Class Fields

Should we require the user to declare all class fields first?

Example:

class Foo:
    a: int32
    b: int32
    def __init__(self, a: int32, b: int32):
        self.a = a
        self.b = b

Subtyping

Do we allow subtyping? Or is parametric polymorphism enough? If subtyping is allowed, we might need virtual method table.

Example where parametric polymorphism is not enough:

class Base:
    def foo();
        return 1

class Foo(Base):
    def foo();
        return 2

def run_all(l: list[Base]):
    return [x.foo() for x in l]

run_all([Base(), Foo()])

Union Type

As function overloading is not possible, should we allow union type, and const evaluate all the type checks after monomorphization?

Example:

def foo(x: Union[int, bool]):
    if type(x) == int:
        return x
    else:
        return 1 if x else 0

Function Pointers

  • Lambda with no capturing are treated as normal functions.
  • Lambda with capturing: a structure would be created to store pointers to captured variables, and the lambda would be a method of the struct. (Note: Storing pointers to meet the binding behavior of Python lambda)
  • Method: implemented with fat pointer, i.e. function pointer + object pointer. Subject to lifetime rules.