TC-3 Samples
Binding is relating a name use to its definition.
Warning
You must strictly follow the format specified below when displaying binding addresses. Addresses must be displayed right after declarations or usages identifiers:
identifier /* 0x55bbe7bdc230 */
Display addresses only when needed. Look closely at the samples.
let
var me := 0
in
me
end
$ tc -XbBA me.tig
/* == Abstract Syntax Tree. == */
function _main /* 0x557fc1caa970 */() =
(
let
var me /* 0x557fc1caac90 */ := 0
in
me /* 0x557fc1caac90 */
end;
()
)
$ echo $?
0
This is harder when there are several occurrences of the same name. Note that primitive types are accepted, but have no pre-declaration, contrary to primitive functions.
let
var me := 0
function id(me : int) : int = me
in
print_int(me)
end
$ tc -bBA meme.tig
/* == Abstract Syntax Tree. == */
primitive print /* 0x560824c3af80 */(string /* 0x560824c386c0 */ : string /* 0 */)
primitive print_err /* 0x560824c39ea0 */(string /* 0x560824c3ad10 */ : string /* 0 */)
primitive print_int /* 0x560824c4ba70 */(int /* 0x560824c4b9f0 */ : int /* 0 */)
primitive flush /* 0x560824c3ac10 */()
primitive getchar /* 0x560824c3a850 */() : string /* 0 */
primitive ord /* 0x560824c3ad90 */(string /* 0x560824c38d10 */ : string /* 0 */) : int /* 0 */
primitive chr /* 0x560824c3c8e0 */(code /* 0x560824c3aed0 */ : int /* 0 */) : string /* 0 */
primitive size /* 0x560824c4e510 */(string /* 0x560824c4c5b0 */ : string /* 0 */) : int /* 0 */
primitive streq /* 0x560824c4e8a0 */(s1 /* 0x560824c4e630 */ : string /* 0 */, s2 /* 0x560824c4e7b0 */ : string /* 0 */) : int /* 0 */
primitive strcmp /* 0x560824c4ec40 */(s1 /* 0x560824c4e9e0 */ : string /* 0 */, s2 /* 0x560824c4eb50 */ : string /* 0 */) : int /* 0 */
primitive substring /* 0x560824c4d1f0 */(string /* 0x560824c4ed80 */ : string /* 0 */, start /* 0x560824c4ef40 */ : int /* 0 */, length /* 0x560824c4d100 */ : int /* 0 */) : string /* 0 */
primitive concat /* 0x560824c4d5d0 */(fst /* 0x560824c4d380 */ : string /* 0 */, snd /* 0x560824c4d4e0 */ : string /* 0 */) : string /* 0 */
primitive not /* 0x560824c4d8d0 */(boolean /* 0x560824c4d760 */ : int /* 0 */) : int /* 0 */
primitive exit /* 0x560824c4dae0 */(status /* 0x560824c4da60 */ : int /* 0 */)
function _main /* 0x560824c4dd70 */() =
(
let
var me /* 0x560824c38c90 */ := 0
function id /* 0x560824c38830 */(me /* 0x560824c38970 */ : int /* 0 */) : int /* 0 */ =
me /* 0x560824c38970 */
in
print_int /* 0x560824c4ba70 */(me /* 0x560824c38c90 */)
end;
()
)
$ echo $?
0
TC-3 is in charge of incorrect uses of the names, such as undefined names,
me
$ tc -bBA nome.tig
nome.tig:1.1-2: undeclared variable: me
$ echo $?
4
or redefined names.
let
type me = {}
type me = {}
function twice(a: int, a: int) : int = a + a
in
me {} = me {}
end
$ tc -bBA tome.tig
tome.tig:3.3-14: redefinition: me
tome.tig:2.3-14: first definition
tome.tig:4.25-31: redefinition: a
tome.tig:4.18-23: first definition
$ echo $?
4
In addition to binding names, --bindings-compute
is also in charge
of binding the break
to their corresponding loop construct.
let var x := 0 in
while 1 do
(
for i := 0 to 10 do
(
x := x + i;
if x >= 42 then
break
);
if x >= 51 then
break
)
end
$ tc -XbBA breaks-in-embedded-loops.tig
/* == Abstract Syntax Tree. == */
function _main /* 0x55f6569076c0 */() =
(
let
var x /* 0x55f656909f80 */ := 0
in
while /* 0x55f65690aa30 */ 1 do
(
for /* 0x55f65693c050 */ i /* 0x55f656907830 */ := 0 to 10 do
(
x /* 0x55f656909f80 */ := x /* 0x55f656909f80 */ + i /* 0x55f656907830 */;
if x /* 0x55f656909f80 */ >= 42
then break /* 0x55f65693c050 */
else ()
);
if x /* 0x55f656909f80 */ >= 51
then break /* 0x55f65690aa30 */
else ()
)
end;
()
)
$ echo $?
0
break
$ tc -b break.tig
break.tig:1.1-5: `break' outside any loop
$ echo $?
4
Embedded loops show that there is scoping for breaks. Beware that there are places, apparently inside loops, where breaks make no sense.
Although it is a matter of definitions and uses of names, record members are not bound here, because it is easier to implement during type checking. Likewise, duplicate fields are to be reported during type checking.
let
type box = { value : int }
type dup = { value : int, value : string }
var box := box { value = 51 }
in
box.head
end
$ tc -XbBA box.tig
/* == Abstract Syntax Tree. == */
function _main /* 0x55ddc7e43970 */() =
(
let
type box /* 0x55ddc7e458a0 */ = { value : int /* 0 */ }
type dup /* 0x55ddc7e595e0 */ = {
value : int /* 0 */,
value : string /* 0 */
}
var box /* 0x55ddc7e43c90 */ := box /* 0x55ddc7e458a0 */ { value = 51 }
in
box /* 0x55ddc7e43c90 */.head
end;
()
)
$ echo $?
0
$ tc -T box.tig
box.tig:3.33-46: identifier multiply defined: value
box.tig:6.3-10: invalid field: head
$ echo $?
5
But apart from these field-specific checks delayed at TC-4, TC-3 should report other name-related errors. In particular, a field with an invalid type name is a binding error (related to the field’s type, not the field itself) to be reported at TC-3.
let
type rec = { a : unknown }
in
rec { a = 42 }
end
$ tc -XbBA unknown-field-type.tig
unknown-field-type.tig:2.20-26: undeclared type: unknown
$ echo $?
4
Likewise, class members (both attributes and methods) are not to be bound at TC-3, but at the type-checking stage (see TC-4, Type Checking). Therefore, no bindings are to be displayed in regards to object at TC-3.
let
type C = class {}
var c := new C
in
c.missing_method();
c.missing_attribute
end
$ tc -X --object-bindings-compute -BA bad-member-bindings.tig
/* == Abstract Syntax Tree. == */
function _main /* 0x5587675ac970 */() =
(
let
type C /* 0x5587675af620 */ =
class extends Object /* 0 */
{
}
var c /* 0x5587675acc90 */ := new C /* 0x5587675af620 */
in
(
c /* 0x5587675acc90 */.missing_method();
c /* 0x5587675acc90 */.missing_attribute
)
end;
()
)
$ echo $?
0
$ tc --object-types-compute bad-member-bindings.tig
bad-member-bindings.tig:5.3-20: unknown method: missing_method
bad-member-bindings.tig:6.3-21: unknown attribute: missing_attribute
$ echo $?
5
Concerning the super class type, the compiler should just check that this type exists in the environment at TC-3, Bindings. Other checks are left to TC-4 (see TC-4 Samples).
let
/* Super class doesn't exist. */
class Z extends Ghost {}
in
end
$ tc -X --object-bindings-compute -BA missing-super-class.tig
missing-super-class.tig:3.19-23: undeclared type: Ghost
$ echo $?
4
The self
identifier is not a reserved keyword, it can be
used to name variables, functions and types, even inside a class.
When used inside a method, self
refers to the current
instance of the object if no variable named self
is
present in the current scope.
let
class self
{
var self : self := nil
method self() : self = self.self
}
var self : self := new self
class foo
{
var self := self
method self(self : self) : self = self.self
}
in
end
$ tc -X --object-bindings-compute -BA self.tig
/* == Abstract Syntax Tree. == */
function _main /* 0x5623ef962d10 */() =
(
let
type self /* 0x5623ef963210 */ =
class extends Object /* 0 */
{
var self : self /* 0x5623ef963210 */ := nil
method self() : self /* 0x5623ef963210 */ =
self /* 0 */.self
}
var self /* 0x5623ef960970 */ : self /* 0x5623ef963210 */ := new self /* 0x5623ef963210 */
type foo /* 0x5623ef961780 */ =
class extends Object /* 0 */
{
var self := self /* 0x5623ef960970 */
method self(self /* 0x5623ef9606c0 */ : self /* 0x5623ef963210 */) : self /* 0x5623ef963210 */ =
self /* 0x5623ef9606c0 */.self
}
in
()
end;
()
)
$ echo $?
0