294f76acd4
There were significant overlaps in the names of data structures between the compiler and the language that made it necessary to litter the code with P4Lang.xxxx. This refactor removes that requirement in most places (Parser is ambiguous wherever TreeSitter is used -- cannot avoid that!) Signed-off-by: Will Hawkins <hawkinsw@obs.cr>
557 lines
20 KiB
Swift
557 lines
20 KiB
Swift
// p4rse, Copyright 2026, Will Hawkins
|
|
//
|
|
// This file is part of p4rse.
|
|
//
|
|
// This file 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 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// 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 <https://www.gnu.org/licenses/>.
|
|
|
|
import Common
|
|
import Foundation
|
|
import Macros
|
|
import P4Lang
|
|
import P4Runtime
|
|
import SwiftTreeSitter
|
|
import Testing
|
|
import TreeSitter
|
|
import TreeSitterP4
|
|
|
|
@testable import P4Compiler
|
|
|
|
@Test func test_field_access() async throws {
|
|
let simple_parser_declaration = """
|
|
parser main_parser() {
|
|
state start {
|
|
bool where_to = ts.yesno;
|
|
transition select (where_to) {
|
|
true: accept;
|
|
false: reject;
|
|
};
|
|
}
|
|
};
|
|
"""
|
|
var test_declarations = StaticVarValueScopes().enter()
|
|
let fields = P4StructFields([
|
|
P4StructFieldIdentifier(name: "yesno", withType: P4QualifiedType(P4Boolean())),
|
|
P4StructFieldIdentifier(name: "count", withType: P4QualifiedType(P4Int())),
|
|
])
|
|
let struct_type = P4Struct(withName: Identifier(name: "Testing"), andFields: fields)
|
|
test_declarations = test_declarations.declare(
|
|
identifier: Identifier(name: "ts"), withValue: (P4QualifiedType(struct_type), .none))
|
|
|
|
var test_values = VarValueScopes().enter()
|
|
test_values = test_values.declare(
|
|
identifier: Identifier(name: "ts"),
|
|
withValue: P4Value(
|
|
P4StructValue(
|
|
withType: struct_type,
|
|
andInitializers: [
|
|
P4Value(P4BooleanValue(withValue: true)),
|
|
P4Value(P4IntValue(withValue: 5)),
|
|
])))
|
|
|
|
let program = try #UseOkResult(
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration, withGlobalInstances: test_declarations))
|
|
let runtime = try #UseOkResult(
|
|
P4Runtime.Runtime<InstantiatedParserState, P4Lang.ParserValue>.create(
|
|
program: program, withGlobalValues: test_values))
|
|
let (state_result, _) = try! #UseOkResult(runtime.run())
|
|
#expect(state_result == P4Lang.accept)
|
|
}
|
|
|
|
@Test func test_field_access_declared() async throws {
|
|
let simple_parser_declaration = """
|
|
parser main_parser() {
|
|
state start {
|
|
Testing ts;
|
|
ts.yesno = true;
|
|
bool where_to = ts.yesno;
|
|
transition select (where_to) {
|
|
true: accept;
|
|
false: reject;
|
|
};
|
|
}
|
|
};
|
|
"""
|
|
var test_types = TypeTypeScopes().enter()
|
|
let fields = P4StructFields([
|
|
P4StructFieldIdentifier(name: "yesno", withType: P4QualifiedType(P4Boolean())),
|
|
P4StructFieldIdentifier(name: "count", withType: P4QualifiedType(P4Int())),
|
|
])
|
|
let struct_type = P4Struct(withName: Identifier(name: "Testing"), andFields: fields)
|
|
test_types = test_types.declare(identifier: Identifier(name: "Testing"), withValue: struct_type)
|
|
|
|
let program = try #UseOkResult(
|
|
SpecialCompilers.ProgramCompiler.Compile(
|
|
simple_parser_declaration, withGlobalInstances: .none, withGlobalTypes: test_types))
|
|
let runtime = try #UseOkResult(
|
|
P4Runtime.Runtime<InstantiatedParserState, P4Lang.ParserValue>.create(program: program))
|
|
let (state_result, _) = try! #UseOkResult(runtime.run())
|
|
#expect(state_result == P4Lang.accept)
|
|
}
|
|
|
|
@Test func test_field_access_declared2() async throws {
|
|
let simple_parser_declaration = """
|
|
parser main_parser() {
|
|
state start {
|
|
Testing ts;
|
|
ts.yesno = true;
|
|
ts.count = 5;
|
|
bool where_to = ts.yesno;
|
|
transition select (ts.count == 5) {
|
|
true: accept;
|
|
false: reject;
|
|
};
|
|
}
|
|
};
|
|
"""
|
|
var test_declarations = TypeTypeScopes().enter()
|
|
let fields = P4StructFields([
|
|
P4StructFieldIdentifier(name: "yesno", withType: P4QualifiedType(P4Boolean())),
|
|
P4StructFieldIdentifier(name: "count", withType: P4QualifiedType(P4Int())),
|
|
])
|
|
let struct_type = P4Struct(withName: Identifier(name: "Testing"), andFields: fields)
|
|
test_declarations = test_declarations.declare(
|
|
identifier: Identifier(name: "Testing"), withValue: struct_type)
|
|
|
|
let program = try #UseOkResult(
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration, withGlobalInstances: .none, withGlobalTypes: test_declarations))
|
|
|
|
let runtime = try #UseOkResult(
|
|
P4Runtime.Runtime<InstantiatedParserState, P4Lang.ParserValue>.create(program: program))
|
|
let (state_result, _) = try! #UseOkResult(runtime.run())
|
|
#expect(state_result == P4Lang.accept)
|
|
}
|
|
|
|
@Test func test_field_access_opp() async throws {
|
|
let simple_parser_declaration = """
|
|
parser main_parser() {
|
|
state start {
|
|
bool where_to = ts.yesno;
|
|
transition select (where_to) {
|
|
true: accept;
|
|
false: reject;
|
|
};
|
|
}
|
|
};
|
|
"""
|
|
var test_declarations = StaticVarValueScopes().enter()
|
|
let fields = P4StructFields([
|
|
P4StructFieldIdentifier(name: "yesno", withType: P4QualifiedType(P4Boolean())),
|
|
P4StructFieldIdentifier(name: "count", withType: P4QualifiedType(P4Int())),
|
|
])
|
|
let struct_type = P4Struct(withName: Identifier(name: "Testing"), andFields: fields)
|
|
test_declarations = test_declarations.declare(
|
|
identifier: Identifier(name: "ts"), withValue: (P4QualifiedType(struct_type), .none))
|
|
|
|
var test_values = VarValueScopes().enter()
|
|
test_values = test_values.declare(
|
|
identifier: Identifier(name: "ts"),
|
|
withValue: P4Value(
|
|
P4StructValue(
|
|
withType: struct_type,
|
|
andInitializers: [
|
|
P4Value(P4BooleanValue(withValue: false)),
|
|
P4Value(P4IntValue(withValue: 5)),
|
|
])))
|
|
|
|
let program = try #UseOkResult(
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration, withGlobalInstances: test_declarations))
|
|
let runtime = try #UseOkResult(
|
|
P4Runtime.Runtime<InstantiatedParserState, P4Lang.ParserValue>.create(
|
|
program: program, withGlobalValues: test_values))
|
|
let (state_result, _) = try! #UseOkResult(runtime.run())
|
|
#expect(state_result == P4Lang.reject)
|
|
}
|
|
|
|
@Test func test_field_access2() async throws {
|
|
let simple_parser_declaration = """
|
|
parser main_parser() {
|
|
state start {
|
|
transition select (ts.count == 5) {
|
|
true: accept;
|
|
false: reject;
|
|
};
|
|
}
|
|
};
|
|
"""
|
|
var test_declarations = StaticVarValueScopes().enter()
|
|
let fields = P4StructFields([
|
|
P4StructFieldIdentifier(name: "yesno", withType: P4QualifiedType(P4Boolean())),
|
|
P4StructFieldIdentifier(name: "count", withType: P4QualifiedType(P4Int())),
|
|
])
|
|
let struct_type = P4Struct(withName: Identifier(name: "Testing"), andFields: fields)
|
|
test_declarations = test_declarations.declare(
|
|
identifier: Identifier(name: "ts"), withValue: (P4QualifiedType(struct_type), .none))
|
|
|
|
var test_values = VarValueScopes().enter()
|
|
test_values = test_values.declare(
|
|
identifier: Identifier(name: "ts"),
|
|
withValue: P4Value(
|
|
P4StructValue(
|
|
withType: struct_type,
|
|
andInitializers: [
|
|
P4Value(P4BooleanValue(withValue: true)),
|
|
P4Value(P4IntValue(withValue: 5)),
|
|
])))
|
|
|
|
let program = try #UseOkResult(
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration, withGlobalInstances: test_declarations))
|
|
let runtime = try #UseOkResult(
|
|
P4Runtime.Runtime<InstantiatedParserState, P4Lang.ParserValue>.create(
|
|
program: program, withGlobalValues: test_values))
|
|
let (state_result, _) = try! #UseOkResult(runtime.run())
|
|
#expect(state_result == P4Lang.accept)
|
|
}
|
|
|
|
@Test func test_field_access2_opp() async throws {
|
|
let simple_parser_declaration = """
|
|
parser main_parser() {
|
|
state start {
|
|
transition select (ts.count == 5) {
|
|
true: accept;
|
|
false: reject;
|
|
};
|
|
}
|
|
};
|
|
"""
|
|
var test_declarations = StaticVarValueScopes().enter()
|
|
let fields = P4StructFields([
|
|
P4StructFieldIdentifier(name: "yesno", withType: P4QualifiedType(P4Boolean())),
|
|
P4StructFieldIdentifier(name: "count", withType: P4QualifiedType(P4Int())),
|
|
])
|
|
let struct_type = P4Struct(withName: Identifier(name: "Testing"), andFields: fields)
|
|
test_declarations = test_declarations.declare(
|
|
identifier: Identifier(name: "ts"), withValue: (P4QualifiedType(struct_type), .none))
|
|
|
|
var test_values = VarValueScopes().enter()
|
|
test_values = test_values.declare(
|
|
identifier: Identifier(name: "ts"),
|
|
withValue: P4Value(
|
|
P4StructValue(
|
|
withType: struct_type,
|
|
andInitializers: [
|
|
P4Value(P4BooleanValue(withValue: true)),
|
|
P4Value(P4IntValue(withValue: 8)),
|
|
])))
|
|
|
|
let program = try #UseOkResult(
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration, withGlobalInstances: test_declarations))
|
|
let runtime = try #UseOkResult(
|
|
P4Runtime.Runtime<InstantiatedParserState, P4Lang.ParserValue>.create(
|
|
program: program, withGlobalValues: test_values))
|
|
let (state_result, _) = try! #UseOkResult(runtime.run())
|
|
#expect(state_result == P4Lang.reject)
|
|
}
|
|
|
|
@Test func test_field_access_nested() async throws {
|
|
let simple_parser_declaration = """
|
|
parser main_parser() {
|
|
state start {
|
|
int where_to = ts.ty.count;
|
|
transition select (where_to == 5) {
|
|
true: accept;
|
|
false: reject;
|
|
};
|
|
}
|
|
};
|
|
"""
|
|
var test_declarations = StaticVarValueScopes().enter()
|
|
|
|
let ty_fields = P4StructFields([
|
|
P4StructFieldIdentifier(name: "yesno", withType: P4QualifiedType(P4Boolean())),
|
|
P4StructFieldIdentifier(name: "count", withType: P4QualifiedType(P4Int())),
|
|
])
|
|
let ty_struct_type = P4Struct(withName: Identifier(name: "nested"), andFields: ty_fields)
|
|
|
|
let ts_fields = P4StructFields([
|
|
P4StructFieldIdentifier(name: "ty", withType: P4QualifiedType(ty_struct_type))
|
|
])
|
|
let ts_struct_type = P4Struct(withName: Identifier(name: "outer"), andFields: ts_fields)
|
|
|
|
test_declarations = test_declarations.declare(
|
|
identifier: Identifier(name: "ts"), withValue: (P4QualifiedType(ts_struct_type), .none))
|
|
|
|
var test_values = VarValueScopes().enter()
|
|
|
|
test_values = test_values.declare(
|
|
identifier: Identifier(name: "ts"),
|
|
withValue: P4Value(
|
|
P4StructValue(
|
|
withType: ts_struct_type,
|
|
andInitializers: [
|
|
P4Value(
|
|
P4StructValue(
|
|
withType: ty_struct_type,
|
|
andInitializers: [
|
|
P4Value(P4BooleanValue(withValue: true)),
|
|
P4Value(P4IntValue(withValue: 5)),
|
|
]))
|
|
])))
|
|
let program = try #UseOkResult(
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration, withGlobalInstances: test_declarations))
|
|
let runtime = try #UseOkResult(
|
|
P4Runtime.Runtime<InstantiatedParserState, P4Lang.ParserValue>.create(
|
|
program: program, withGlobalValues: test_values))
|
|
let (state_result, _) = try! #UseOkResult(runtime.run())
|
|
#expect(state_result == P4Lang.accept)
|
|
}
|
|
|
|
@Test func test_field_write() async throws {
|
|
let simple_parser_declaration = """
|
|
parser main_parser() {
|
|
state start {
|
|
ts.yesno = true;
|
|
bool where_to = ts.yesno;
|
|
transition select (where_to) {
|
|
true: accept;
|
|
false: reject;
|
|
};
|
|
}
|
|
};
|
|
"""
|
|
var test_declarations = StaticVarValueScopes().enter()
|
|
let fields = P4StructFields([
|
|
P4StructFieldIdentifier(name: "yesno", withType: P4QualifiedType(P4Boolean())),
|
|
P4StructFieldIdentifier(name: "count", withType: P4QualifiedType(P4Int())),
|
|
])
|
|
let struct_type = P4Struct(withName: Identifier(name: "Testing"), andFields: fields)
|
|
test_declarations = test_declarations.declare(
|
|
identifier: Identifier(name: "ts"), withValue: (P4QualifiedType(struct_type), .none))
|
|
|
|
var test_values = VarValueScopes().enter()
|
|
test_values = test_values.declare(
|
|
identifier: Identifier(name: "ts"),
|
|
withValue: P4Value(
|
|
P4StructValue(
|
|
withType: struct_type,
|
|
andInitializers: [
|
|
P4Value(P4BooleanValue(withValue: false)),
|
|
P4Value(P4IntValue(withValue: 5)),
|
|
])))
|
|
|
|
let program = try #UseOkResult(
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration, withGlobalInstances: test_declarations))
|
|
let runtime = try #UseOkResult(
|
|
P4Runtime.Runtime<InstantiatedParserState, P4Lang.ParserValue>.create(
|
|
program: program, withGlobalValues: test_values))
|
|
let (state_result, _) = try! #UseOkResult(runtime.run())
|
|
#expect(state_result == P4Lang.accept)
|
|
}
|
|
|
|
@Test func test_field_write_invalid_type() async throws {
|
|
let simple_parser_declaration = """
|
|
parser main_parser() {
|
|
state start {
|
|
ts.yesno = 5;
|
|
transition accept;
|
|
}
|
|
};
|
|
"""
|
|
var test_declarations = StaticVarValueScopes().enter()
|
|
let fields = P4StructFields([
|
|
P4StructFieldIdentifier(name: "yesno", withType: P4QualifiedType(P4Boolean())),
|
|
P4StructFieldIdentifier(name: "count", withType: P4QualifiedType(P4Int())),
|
|
])
|
|
let struct_type = P4Struct(withName: Identifier(name: "Testing"), andFields: fields)
|
|
test_declarations = test_declarations.declare(
|
|
identifier: Identifier(name: "ts"), withValue: (P4QualifiedType(struct_type), .none))
|
|
|
|
#expect(
|
|
#RequireErrorResult(
|
|
Error(
|
|
withMessage:
|
|
"{49, 13}: Failed to parse a statement element: {49, 8}: Cannot assign value of type Int (width: Infinite) to field yesno of type Boolean"
|
|
),
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration, withGlobalInstances: test_declarations))
|
|
)
|
|
}
|
|
|
|
@Test func test_field_write_nested() async throws {
|
|
let simple_parser_declaration = """
|
|
parser main_parser() {
|
|
state start {
|
|
ts.ty.count = 5;
|
|
int where_to = ts.ty.count;
|
|
transition select (where_to == 5) {
|
|
true: accept;
|
|
false: reject;
|
|
};
|
|
}
|
|
};
|
|
"""
|
|
var test_declarations = StaticVarValueScopes().enter()
|
|
|
|
let ty_fields = P4StructFields([
|
|
P4StructFieldIdentifier(name: "yesno", withType: P4QualifiedType(P4Boolean())),
|
|
P4StructFieldIdentifier(name: "count", withType: P4QualifiedType(P4Int())),
|
|
])
|
|
let ty_struct_type = P4Struct(withName: Identifier(name: "nested"), andFields: ty_fields)
|
|
|
|
let ts_fields = P4StructFields([
|
|
P4StructFieldIdentifier(name: "ty", withType: P4QualifiedType(ty_struct_type))
|
|
])
|
|
let ts_struct_type = P4Struct(withName: Identifier(name: "outer"), andFields: ts_fields)
|
|
|
|
test_declarations = test_declarations.declare(
|
|
identifier: Identifier(name: "ts"), withValue: (P4QualifiedType(ts_struct_type), .none))
|
|
|
|
var test_values = VarValueScopes().enter()
|
|
|
|
test_values = test_values.declare(
|
|
identifier: Identifier(name: "ts"),
|
|
withValue: P4Value(
|
|
P4StructValue(
|
|
withType: ts_struct_type,
|
|
andInitializers: [
|
|
P4Value(
|
|
P4StructValue(
|
|
withType: ty_struct_type,
|
|
andInitializers: [
|
|
P4Value(P4BooleanValue(withValue: true)),
|
|
P4Value(P4IntValue(withValue: 7)),
|
|
]))
|
|
])))
|
|
let program = try #UseOkResult(
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration, withGlobalInstances: test_declarations))
|
|
let runtime = try #UseOkResult(
|
|
P4Runtime.Runtime<InstantiatedParserState, P4Lang.ParserValue>.create(
|
|
program: program, withGlobalValues: test_values))
|
|
let (state_result, _) = try! #UseOkResult(runtime.run())
|
|
#expect(state_result == P4Lang.accept)
|
|
}
|
|
|
|
@Test func test_field_write_nested2() async throws {
|
|
let simple_parser_declaration = """
|
|
parser main_parser() {
|
|
state start {
|
|
ts.ty.count = 3;
|
|
ts.ty.count = 5;
|
|
int where_to = ts.ty.count;
|
|
transition select (where_to == 5) {
|
|
true: accept;
|
|
false: reject;
|
|
};
|
|
}
|
|
};
|
|
"""
|
|
var test_declarations = StaticVarValueScopes().enter()
|
|
|
|
let ty_fields = P4StructFields([
|
|
P4StructFieldIdentifier(name: "yesno", withType: P4QualifiedType(P4Boolean())),
|
|
P4StructFieldIdentifier(name: "count", withType: P4QualifiedType(P4Int())),
|
|
])
|
|
let ty_struct_type = P4Struct(withName: Identifier(name: "nested"), andFields: ty_fields)
|
|
|
|
let ts_fields = P4StructFields([
|
|
P4StructFieldIdentifier(name: "ty", withType: P4QualifiedType(ty_struct_type))
|
|
])
|
|
let ts_struct_type = P4Struct(withName: Identifier(name: "outer"), andFields: ts_fields)
|
|
|
|
test_declarations = test_declarations.declare(
|
|
identifier: Identifier(name: "ts"), withValue: (P4QualifiedType(ts_struct_type), .none))
|
|
|
|
var test_values = VarValueScopes().enter()
|
|
|
|
test_values = test_values.declare(
|
|
identifier: Identifier(name: "ts"),
|
|
withValue: P4Value(
|
|
P4StructValue(
|
|
withType: ts_struct_type,
|
|
andInitializers: [
|
|
P4Value(
|
|
P4StructValue(
|
|
withType: ty_struct_type,
|
|
andInitializers: [
|
|
P4Value(P4BooleanValue(withValue: true)),
|
|
P4Value(P4IntValue(withValue: 7)),
|
|
]))
|
|
])))
|
|
let program = try #UseOkResult(
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration, withGlobalInstances: test_declarations))
|
|
let runtime = try #UseOkResult(
|
|
P4Runtime.Runtime<InstantiatedParserState, P4Lang.ParserValue>.create(
|
|
program: program, withGlobalValues: test_values))
|
|
let (state_result, _) = try! #UseOkResult(runtime.run())
|
|
#expect(state_result == P4Lang.accept)
|
|
}
|
|
|
|
@Test func test_field_write_nested_invalid_type() async throws {
|
|
let simple_parser_declaration = """
|
|
parser main_parser() {
|
|
state start {
|
|
ts.ty.count = false;
|
|
int where_to = ts.ty.count;
|
|
transition select (where_to == 5) {
|
|
true: accept;
|
|
false: reject;
|
|
};
|
|
}
|
|
};
|
|
"""
|
|
var test_declarations = StaticVarValueScopes().enter()
|
|
|
|
let ty_fields = P4StructFields([
|
|
P4StructFieldIdentifier(name: "yesno", withType: P4QualifiedType(P4Boolean())),
|
|
P4StructFieldIdentifier(name: "count", withType: P4QualifiedType(P4Int())),
|
|
])
|
|
let ty_struct_type = P4Struct(withName: Identifier(name: "nested"), andFields: ty_fields)
|
|
|
|
let ts_fields = P4StructFields([
|
|
P4StructFieldIdentifier(name: "ty", withType: P4QualifiedType(ty_struct_type))
|
|
])
|
|
let ts_struct_type = P4Struct(withName: Identifier(name: "outer"), andFields: ts_fields)
|
|
|
|
test_declarations = test_declarations.declare(
|
|
identifier: Identifier(name: "ts"), withValue: (P4QualifiedType(ts_struct_type), .none))
|
|
|
|
#expect(
|
|
#RequireErrorResult(
|
|
Error(
|
|
withMessage:
|
|
"{49, 20}: Failed to parse a statement element: {49, 11}: Cannot assign value of type Boolean to field count of type Int (width: Infinite)"
|
|
),
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration, withGlobalInstances: test_declarations))
|
|
)
|
|
|
|
}
|
|
|
|
@Test func test_field_read_equality_invalid_type() async throws {
|
|
let simple_parser_declaration = """
|
|
parser main_parser() {
|
|
state start {
|
|
transition select (ts.yesno == "testing") {
|
|
true: accept;
|
|
false: reject;
|
|
};
|
|
}
|
|
};
|
|
"""
|
|
var test_declarations = StaticVarValueScopes().enter()
|
|
let fields = P4StructFields([
|
|
P4StructFieldIdentifier(name: "yesno", withType: P4QualifiedType(P4Boolean())),
|
|
P4StructFieldIdentifier(name: "count", withType: P4QualifiedType(P4Int())),
|
|
])
|
|
let struct_type = P4Struct(withName: Identifier(name: "Testing"), andFields: fields)
|
|
test_declarations = test_declarations.declare(
|
|
identifier: Identifier(name: "ts"), withValue: (P4QualifiedType(struct_type), .none))
|
|
|
|
#expect(
|
|
#RequireErrorResult(
|
|
Error(
|
|
withMessage:
|
|
"{68, 21}: Could not parse transition select expression selector expression: Types of values used with binary expression are not the same"
|
|
),
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration, withGlobalInstances: test_declarations))
|
|
)
|
|
}
|