diff options
Diffstat (limited to 'docs/wiki_src/coding')
-rw-r--r-- | docs/wiki_src/coding/IO.md | 29 | ||||
-rw-r--r-- | docs/wiki_src/coding/REPL.md | 100 | ||||
-rw-r--r-- | docs/wiki_src/coding/combinators.md | 0 | ||||
-rw-r--r-- | docs/wiki_src/coding/compilation.md | 44 | ||||
-rw-r--r-- | docs/wiki_src/coding/currying.md | 13 | ||||
-rw-r--r-- | docs/wiki_src/coding/data-structures.md | 111 | ||||
-rw-r--r-- | docs/wiki_src/coding/examples.md | 0 | ||||
-rw-r--r-- | docs/wiki_src/coding/laziness.md | 52 | ||||
-rw-r--r-- | docs/wiki_src/coding/meta-programming.md | 72 | ||||
-rw-r--r-- | docs/wiki_src/coding/mixfix.md | 33 | ||||
-rw-r--r-- | docs/wiki_src/coding/performance.md | 0 | ||||
-rw-r--r-- | docs/wiki_src/coding/prefix.md | 25 | ||||
-rw-r--r-- | docs/wiki_src/coding/recursion.md | 78 | ||||
-rw-r--r-- | docs/wiki_src/coding/style.md | 36 | ||||
-rw-r--r-- | docs/wiki_src/coding/test-driven-development.md | 42 | ||||
-rw-r--r-- | docs/wiki_src/coding/typing.md | 0 | ||||
-rw-r--r-- | docs/wiki_src/coding/uniform-function-call-syntax.md | 0 |
17 files changed, 635 insertions, 0 deletions
diff --git a/docs/wiki_src/coding/IO.md b/docs/wiki_src/coding/IO.md new file mode 100644 index 0000000..974f98e --- /dev/null +++ b/docs/wiki_src/coding/IO.md @@ -0,0 +1,29 @@ +# IO + +Bruijn supports a variant of John Tromp's monadic IO[^1]. + +Every program's `main`{.bruijn} function has an additional abstraction +that gets applied with a lazy list of input bytes. These bytes are +encoded as the syntactic sugar encoding of binary numbers, which can be +manipulated with [`std/Number/Binary`](/std/Number_Binary.bruijn.html). + +You can use [`std/Monad`](/std/Monad.bruijn.html) to interact with the +input monadically, or simply use [`std/List`](/std/List.bruijn.html) +operations to work with the input as a normal list. + +## Example + +``` bruijn +:import std/List . + +# reverse the input list +main [<~>0] +``` + +``` bash +$ printf "tacocat" | bruijn reverse.bruijn +tacocat +``` + +[^1]: [Tromp, John. "Functional Bits: Lambda Calculus based Algorithmic + Information Theory." (2023).](https://tromp.github.io/cl/LC.pdf) diff --git a/docs/wiki_src/coding/REPL.md b/docs/wiki_src/coding/REPL.md new file mode 100644 index 0000000..a801ef9 --- /dev/null +++ b/docs/wiki_src/coding/REPL.md @@ -0,0 +1,100 @@ +# REPL + +The REPL is a very helpful feature for functional programming languages +like bruijn. You can use it to continuously test or execute parts of +your code. + +You can start the REPL using `stack run`{.bash} or (if installed) +`bruijn`{.bash}. + +Any valid term will get reduced to normal form after pressing enter. + +## Definitions + +Since everything you type will get evaluated, definitions (compared to +definitions in files) require an equal sign: + +``` bruijn +> id = [0] +> id +[0] +``` + +## Commands + +### `:import`{.bruijn}/`:input`{.bruijn} + +Equivalent to the [respective commands in +files](../introduction/syntax.md#imports). + +``` bruijn +> :import std/Math . +``` + +### `:test`{.bruijn} + +Equivalent to the [test command in +files](../introduction/syntax.md#tests). + +``` bruijn +> :test ([0]) ([[1]]) +ERROR test failed: [0] = [[1]] + reduced to [0] = [[1]] +``` + +### `:watch`{.bruijn} + +`:watch`{.bruijn} re-imports the file automatically after every saved +change. It will rerun any test the watched file contains. + +``` bruijn +> :watch collatz-proof +``` + +This can be very helpful for [test driven +development](test-driven-development.md). + +### `:time`{.bruijn} + +Measures the time from start of reduction to its end (normal form) in +seconds. + +``` bruijn +> :time fac (+30) +0.15 seconds +``` + +### `:blc`{.bruijn} + +Translates both the unreduced and the reduced expression to binary +lambda calculus. Helpful for golfed [compilation](compilation.md). + +``` bruijn +> :blc [0] [0] +0100100010 +0010 +``` + +### `:length`{.bruijn} + +Measures the length of the binary lambda calculus encoding of both the +unreduced and the reduced expression. Helpful for golfed +[compilation](compilation.md). + +``` bruijn +> :blc [0] [0] +0100100010 +0010 +``` + +### `:free`{.bruijn} + +The `free` command frees the current environment including all defined +identifiers and imported files. + +``` bruijn +> id = [0] +> :free +> id +ERROR undefined identifier id +``` diff --git a/docs/wiki_src/coding/combinators.md b/docs/wiki_src/coding/combinators.md new file mode 100644 index 0000000..e69de29 --- /dev/null +++ b/docs/wiki_src/coding/combinators.md diff --git a/docs/wiki_src/coding/compilation.md b/docs/wiki_src/coding/compilation.md new file mode 100644 index 0000000..9863c21 --- /dev/null +++ b/docs/wiki_src/coding/compilation.md @@ -0,0 +1,44 @@ +# Compilation + +Bruijn can be compiled to John Tromp’s binary lambda calculus (BLC). + +BLC uses the following encoding: + +| term | lambda | bruijn | BLC | +|:-------------|:-------|:--------|:-----------------| +| abstraction | λM | `[M]` | 00M | +| application | (MN) | `(M N)` | 01MN | +| bruijn index | i | `i` | 1<sup>i+1</sup>0 | + +There are two modes of compilation: + +- **Bitwise** compiles to BLC and encodes every bit as 1 bit and pads + the last remaining byte: `bruijn -b path` +- **ASCII** compiles to BLC and encodes every bit as 1 ASCII character + (`'0'`/`'1'`): `bruijn -B path` + +## Compilation overhead + +Typical compilation to BLC results in much redundant code, since every +used function gets substituted and translated separately. In +`((+3) + (+4) + (+3))`, for example, `add` gets compiled to BLC two +times, resulting in a redundant overhead of around 3500 bits. + +This is because BLC was never intended for compilation of normal +programs, but mainly as an academic encoding model. This also means that +it’s quite good for writing very expressive and minimal programs +(i.e. obfuscated code golfing, see [John Tromp’s +IOCCC](https://ioccc.org/2012/tromp/hint.html)). + +Most programs, however, won’t be golfed and can result in rather large +compiled programs. While there’s not really any practical need for +compilation aside from golfing, you could still use the +[BLoC](https://github.com/marvinborner/bloc) project to optimize +redundant terms. + +Typical workflow: + +``` bash +$ bruijn -B program.bruijn | bloc --from-blc -i - -o out.bloc +$ cat input | bruijn -E <(bloc --from-bloc -i out.bloc) +``` diff --git a/docs/wiki_src/coding/currying.md b/docs/wiki_src/coding/currying.md new file mode 100644 index 0000000..d84eb0a --- /dev/null +++ b/docs/wiki_src/coding/currying.md @@ -0,0 +1,13 @@ +# Currying + +Lambda calculus naturally supports currying -- that is, only partially +applying a function. In fact *any* function can be applied with *any* +amount of arguments! + +In bruijn, currying is a great way to make functions even more elegant. + +Partially applying the `mul`{.bruijn} function: + +``` bruijn +six [0 (+3)] (mul (+2)) +``` diff --git a/docs/wiki_src/coding/data-structures.md b/docs/wiki_src/coding/data-structures.md new file mode 100644 index 0000000..384ac29 --- /dev/null +++ b/docs/wiki_src/coding/data-structures.md @@ -0,0 +1,111 @@ +# Data structures + +Bruijn's standard library defines several common data structures. + +Relevant blog post: [Data structures in pure lambda +calculus](https://text.marvinborner.de/2023-04-07-01.html). + +## States + +For storing states (i.e. enums), you can use the available libraries and +syntactic sugar. + +### Booleans/bits [`std/Logic`](/std/Logic.bruijn.html) + +- typical [Church + booleans](https://en.wikipedia.org/wiki/Church_encoding#Church_Booleans) + -- fast and reliable +- encoding: `true`{.bruijn}=`[[1]]`{.bruijn}, + `false`{.bruijn}=`[[0]]`{.bruijn} + +### Unary numbers [`std/Number/Unary`](/std/Number_Unary.bruijn.html) + +- `u` suffix for syntactic sugar, e.g. `(+3u)`{.bruijn} +- encoding: `(+4u)`{.bruijn}=`[[(1 (1 (1 (1 0))))]]`{.bruijn} +- typical [Church + numerals](https://en.wikipedia.org/wiki/Church_encoding#Church_numerals), + simple but high space/time complexity +- only positive numbers + +### Binary numbers [`std/Number/Binary`](/std/Number_Binary.bruijn.html) + +- `b` suffix for syntactic sugar, e.g. `(+3b)`{.bruijn} +- encoding: `(+4b)`{.bruijn}=`[[[0 (0 (1 2))]]]`{.bruijn} +- encoding for chars/strings, e.g. `'0'`{.bruijn}=`(+48b)`{.bruijn} +- faster and more compact than unary +- only positive numbers (excluding two's complement) + +### Balanced ternary [`std/Number/Ternary`](/std/Number_Ternary.bruijn.html) + +- default syntactic sugar for numbers (optional suffix `t`), + e.g. `(+3)`{.bruijn} +- encoding: `(+4)`{.bruijn}=`[[[[(1 (1 3))]]]]`, + `(-4)`{.bruijn}=`[[[[(2 (2 3))]]]]`{.bruijn} +- faster and more compact than binary[^1] +- positive and negative numbers + +## Boxes [`std/Box`](/std/Box.bruijn.html) + +Boxes are good for storing single values as immutable object with an +empty/full state. + +Example: + +``` bruijn +a-box <>'a' + +:test (set? a-box) (true) +:test (get 'b' a-box) ('a') +:test (get 'b' empty) ('b') +:test (store! a-box 'b') (<>'b') +``` + +## Pairs [`std/Pair`](/std/Pair.bruijn.html) + +Pairs (tuples) can store any two terms. + +Example: + +``` bruijn +one-two (+1) : (+2) + +:test (^one-two) ((+1)) +:test (~one-two) ((+2)) +:test (inc <$> one-two) ((+2) : (+3)) +:test (uncurry add one-two) ((+3)) +``` + +## Lists [`std/List`](/std/List.bruijn.html) + +Lists are a repeated composition (right-associative) of pairs with a +`empty`{.bruijn} ending symbol `[[1]]`{.bruijn}. They can store any +(heterogeneous) values and are recursively iterable. The call-by-need +reduction order of bruijn allows lazy evaluation (i.e. infinite lists). + +Example: + +``` bruijn +:test (length (take (+3) (repeat (+4)))) ((+3)) +:test (take (+5) (iterate ++‣ (+0))) (((+0) : ((+1) : ((+2) : ((+3) : {}(+4)))))) +:test ((foldr …+… (+0) ((+1) : ((+2) : {}(+3)))) =? (+6)) (true) +``` + +## Strings [`std/String`](/std/String.bruijn.html) + +Strings are just a list of binary encoded bytes. You may use +[`std/List`](/std/List.bruijn.html) in combinatoin with +[`std/Number/Binary`](/std/Binary.bruijn.html) to interact with them. + +Example: + +``` bruijn +:test (lines "abc\ndef") ("abc" : {}"def") +:test ("ab" =? "ab") (true) +``` + +[^1]: [Mogensen, Torben Æ. "An investigation of compact and efficient + number representations in the pure lambda calculus." Perspectives of + System Informatics: 4th International Andrei Ershov Memorial + Conference, PSI 2001 Akademgorodok, Novosibirsk, Russia, July 2--6, + 2001 Revised Papers 4. Springer Berlin Heidelberg, + 2001.](https://doi.org/10.1007/3-540-45575-2_20) diff --git a/docs/wiki_src/coding/examples.md b/docs/wiki_src/coding/examples.md new file mode 100644 index 0000000..e69de29 --- /dev/null +++ b/docs/wiki_src/coding/examples.md diff --git a/docs/wiki_src/coding/laziness.md b/docs/wiki_src/coding/laziness.md new file mode 100644 index 0000000..feec480 --- /dev/null +++ b/docs/wiki_src/coding/laziness.md @@ -0,0 +1,52 @@ +# Laziness + +Due to the call-by-need reduction order of bruijn, several operations +are lazily evaluated (automagically!). + +## Infinite lists + +You can use infinite list generators like `repeat`{.bruijn} or +`iterate`{.bruijn} to lazily interact with list elements. + +``` bruijn +:import std/List . + +:test (length (take (+3) (repeat (+4)))) ((+3)) +:test (take (+5) (iterate ++‣ (+0))) (((+0) : ((+1) : ((+2) : ((+3) : {}(+4)))))) +``` + +## Math + +``` bruijn +# power function +:test ((iterate (…⋅… (+2)) (+1)) !! (+3)) ((+8)) + +# prime numbers +primes nub ((…≠?… (+1)) ∘∘ gcd) (iterate ++‣ (+2)) ⧗ (List Number) + +:test (take (+4) primes) ((+2) : ((+3) : ((+5) : {}(+7)))) + +# fibonacci +fibs head <$> (iterate [~0 : (^0 + ~0)] ((+0) : (+1))) ⧗ (List Number) + +:test (take (+4) primes) ((+0) : ((+1) : ((+1) : {}(+2)))) +``` + +## Optimization + +Laziness can (in some cases) produce huge performance boosts. For +example: + +``` bruijn +# 11 seconds +:time (+10) ** (+500) + +# 0.1 seconds +:time ((+10) ** (+500)) =? (+400) +``` + +This works because a ternary number is just a list of trits which (in +this case) gets recursively generated by the `pow`{.bruijn} function. +The `eq?`{.bruijn} function just throws away the first argument if it's +already clear that the numbers can't be equal (in this case after the +first argument got bigger than `(+400)`{.bruijn}). diff --git a/docs/wiki_src/coding/meta-programming.md b/docs/wiki_src/coding/meta-programming.md new file mode 100644 index 0000000..44f5ea2 --- /dev/null +++ b/docs/wiki_src/coding/meta-programming.md @@ -0,0 +1,72 @@ +# Meta programming + +TODO: more (blog) + +Bruijn has a homoiconic meta encoding similar to Lisp's quoting feature. + +Blog post with more details: [Homoiconic self interpretation of lambda +calculus](https://text.marvinborner.de/2023-09-03-21.html). + +## Encoding + +``` bruijn +`X ⤳ [[[2 (+Xu)]]] +`(M N) ⤳ [[[1 `M `N]]] +`[M] ⤳ [[[0 `M]]] +``` + +Any quoted term gets converted to this encoding: + +``` bruijn +# example quotations +:test (`0) ([[[2 (+0u)]]]) +:test (`[0]) ([[[0 [[[2 (+0u)]]]]]]) +:test (`'0') ([[[0 [[[0 [[[0 [[[1 [[[2 (+0u)]]] [[[1 [[[2 (+0u)]]] [[[1 [[[2 (+0u)]]] [[[1 [[[2 (+0u)]]] [[[1 [[[2 (+1u)]]] [[[1 [[[2 (+1u)]]] [[[1 [[[2 (+0u)]]] [[[1 [[[2 (+0u)]]] [[[2 (+2u)]]]]]]]]]]]]]]]]]]]]]]]]]]]]]]]]]]]]) + +# quotes are nestable! +:test (``0) ([[[0 [[[0 [[[0 [[[1 [[[2 (+2u)]]] [[[0 [[[0 [[[2 (+0u)]]]]]]]]]]]]]]]]]]]]]) +:test (`[0 `0]) ([[[0 [[[1 [[[2 (+0u)]]] [[[0 [[[0 [[[0 [[[1 [[[2 (+2u)]]] [[[0 [[[0 [[[2 (+0u)]]]]]]]]]]]]]]]]]]]]]]]]]]]) +``` + +## Quasiquotation + +Quoted terms can be escaped (*unquoted*) using the comma symbol. +Unquoted terms will be fully evaluated first before getting quoted +again. + +``` bruijn +:test (```,[0]) (``[0]) +:test (`,`,[0]) ([0]) +:test (`[0 `,[0]]) (`[0 [0]]) +``` + +Unquoted De Bruijn indices will get bound to the respective abstraction +outside of its meta encoding. + +``` bruijn +# adds two using normal quotation +add-two `[0 + (+2u)] + +:test (!add-two (+2u)) ((+4u)) + +# adds two using a reaching De Bruijn index +add-two* [`(,0 + (+2u))] + +:test (!(add-two* (+2u))) ((+4u)) +``` + +## Meta library [`std/Meta`](/std/Meta.bruijn.html) + +The meta library enables simple interaction with the meta encoding. + +Examples: + +``` bruijn +# testing equivalence +:test (α-eq? `[0 0] `[0 0]) (true) +:test (α-eq? `α-eq? `α-eq?) (true) + +# BLC length of meta term +:test (length `[0]) ((+4u)) +:test (length `[[1 1]]) ((+12u)) +``` diff --git a/docs/wiki_src/coding/mixfix.md b/docs/wiki_src/coding/mixfix.md new file mode 100644 index 0000000..c85b936 --- /dev/null +++ b/docs/wiki_src/coding/mixfix.md @@ -0,0 +1,33 @@ +# Mixfix + +Mixfix functions allow arbitrary infix operations based on "substitution +holes" by using the `…` symbol in (special character) definitions. The +symbols and terms always need to be delimited by a space character, else +they get interpreted as a [prefix](prefix.md). + +Example: + +``` bruijn +…+… add + +# the "holes" get applied in normal order +:test ((+4) + (+3)) (add (+4) (+3)) +``` + +You can define as many holes as you like. Make sure to place parenthesis +for applications inside substitution holes. + +``` bruijn +{…<$>…|… [[[2 - 1 + 0]]] + +# evaluated as (5 - 2) + 1 = 4 +:test ({ (+5) <$> (+2) | (+1)) ((+4)) +:test ({ ((+3) + (+1)) <$> (+2) | (+1)) ((+4)) +``` + +You can use them as normal functions by writing the identifier +literally: + +``` bruijn +:test (…+… (+4) (+3)) (add (+4) (+3)) +``` diff --git a/docs/wiki_src/coding/performance.md b/docs/wiki_src/coding/performance.md new file mode 100644 index 0000000..e69de29 --- /dev/null +++ b/docs/wiki_src/coding/performance.md diff --git a/docs/wiki_src/coding/prefix.md b/docs/wiki_src/coding/prefix.md new file mode 100644 index 0000000..589e081 --- /dev/null +++ b/docs/wiki_src/coding/prefix.md @@ -0,0 +1,25 @@ +# Prefix + +Prefix functions are symbols written directly in front of another term +(without space). The term gets applied as an argument to the prefix +function. Use [mixfix functions](mixfix.md) if the function has more +than one argument. + +They are defined by the `‣`{.bruijn} suffix. + +Example: + +``` bruijn +# defines a negation prefix function called '-' +-‣ [(+0) - 0] + +# returns 0 - 10 = -10 +:test (-(+10)) ((-10)) +``` + +You can use them as normal functions by writing the identifier +literally: + +``` bruijn +:test (-‣ (+10)) ((-10)) +``` diff --git a/docs/wiki_src/coding/recursion.md b/docs/wiki_src/coding/recursion.md new file mode 100644 index 0000000..68148e5 --- /dev/null +++ b/docs/wiki_src/coding/recursion.md @@ -0,0 +1,78 @@ +# Recursion + +Just as normal lambda calculus, bruijn does *not* support typical +recursion. + +If you want to recursively call a function (or imitate `for`/`while` +loops), you need to use *fixed-point combinators* like `y`{.bruijn} from +[`std/Combinator`](/std/Combinator.bruijn.html). + +Fixed-point combinators have the fascinating property of inducing +recursive behaviour in programming languages without support for +recursion. + +Say we want a function `g`{.bruijn} to be able to call itself. With the +`y`{.bruijn} combinator the following equivalence is obtained: + +``` bruijn + (y g) +⤳ [[1 (0 0)] [1 (0 0)]] g +⤳ [g (0 0)] [g (0 0)] +⤳ g ([g (0 0)] [g (0 0)]) +≡ g (y g) +``` + +With this equivalence, `g`{.bruijn} is able to call itself since its +outer argument is the initial function again. + +Example for using `y`{.bruijn} to find the factorial of 2: + +``` bruijn +# here, `1` is the outer argument (y g) +# `0` is the accumulator (the argument of `factorial`) +g [[=?0 (+1) (0 ⋅ (1 --0))]] + +factorial y g ⧗ Number → Number + +:test ((factorial (+3)) =? (+6)) (true) +``` + +In-the-wild, this could look like this. + +``` bruijn +# 3 abstractions => two arguments +# 2 is recursive call +# 1 is accumulator (+0) +# 0 is argument (list) +length z [[[rec]]] (+0) ⧗ (List a) → Number + rec 0 [[[case-inc]]] case-end + case-inc 5 ++4 1 + case-end 1 +``` + +Also see [coding style](style.md) for other style suggestions. + +## Mutual recurrence relations + +For solving mutual recurrence relations, you can use the *variadic +fixed-point combinator* `y*`{.bruijn} from +[`std/List`](/std/List.bruijn.html). This combinator produces all the +fixed points of a function as an iterable [list](data-structures.md). + +Example `even?`{.bruijn}/`odd?`{.bruijn} implementation using +`y*`{.bruijn}: + +``` bruijn +# the odd? recursive call will be the second argument (1) +g [[[=?0 true (1 --0)]]] + +# the even? recursive call will be the first argument (2) +h [[[=?0 false (2 --0)]]] + +even? head (y* g h) ⧗ Number → Bool + +odd? tail (y* g h) ⧗ Number → Bool +``` + +Read more about this in the blog post [Variadic fixed-point +combinators](https://text.marvinborner.de/2023-06-18-15.html). diff --git a/docs/wiki_src/coding/style.md b/docs/wiki_src/coding/style.md new file mode 100644 index 0000000..d0ec7a9 --- /dev/null +++ b/docs/wiki_src/coding/style.md @@ -0,0 +1,36 @@ +# Coding style + +## Scoping + +## If/else + +redundant + +## Head/tail + +redundant + +## Type signatures + +## Recursion + +[Recursion](recursion.md) should almost always be achieved with the +`y`{.bruijn} or `z`{.bruijn} combinators. + +A common coding style in bruijn's standard library is to use the scoped +`rec` function to indicate recursion. You would then use `n+1` +abstraction around `rec` to indicate `n` arguments and the additionally +induced recursive call. + +Example of the `length`{.bruijn} function for lists: + +``` bruijn +# 3 abstractions => two arguments +# 2 is recursive call +# 1 is accumulator (+0) +# 0 is argument (list) +length z [[[rec]]] (+0) ⧗ (List a) → Number + rec 0 [[[case-inc]]] case-end + case-inc 5 ++4 1 + case-end 1 +``` diff --git a/docs/wiki_src/coding/test-driven-development.md b/docs/wiki_src/coding/test-driven-development.md new file mode 100644 index 0000000..f62c448 --- /dev/null +++ b/docs/wiki_src/coding/test-driven-development.md @@ -0,0 +1,42 @@ +# Test driven development (TDD) + +The suggested technique for bruijn development is the TDD method. When +creating functions, we suggest the following procedure: + +- Write a comment, a type signature, and the head of the function + +``` bruijn +# measures the length of a list +length y [[[rec]]] (+0) ⧗ (List a) → Number +``` + +- Write several tests including all edge cases + +``` bruijn +# measures the length of a list +length y [[[rec]]] (+0) ⧗ (List a) → Number + +:test (length empty) ((+0)) +:test (length "a") ((+1)) +:test (length ({}empty)) ((+1)) +:test (length (empty : {}empty)) ((+2)) +:test (length ("abc")) ((+3)) +``` + +- Finish the implementation until all tests pass (e.g. using the + [`:watch`{.bruijn} repl command](REPL.md#watch)) +- Refactor and clean up the definition + +``` bruijn +# measures the length of a list +length y [[[rec]]] (+0) ⧗ (List a) → Number + rec 0 [[[case-inc]]] case-end + case-inc 5 ++4 1 + case-end 1 + +:test (length empty) ((+0)) +:test (length "a") ((+1)) +:test (length ({}empty)) ((+1)) +:test (length (empty : {}empty)) ((+2)) +:test (length ("abc")) ((+3)) +``` diff --git a/docs/wiki_src/coding/typing.md b/docs/wiki_src/coding/typing.md new file mode 100644 index 0000000..e69de29 --- /dev/null +++ b/docs/wiki_src/coding/typing.md diff --git a/docs/wiki_src/coding/uniform-function-call-syntax.md b/docs/wiki_src/coding/uniform-function-call-syntax.md new file mode 100644 index 0000000..e69de29 --- /dev/null +++ b/docs/wiki_src/coding/uniform-function-call-syntax.md |