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>
403 lines
9.2 KiB
Swift
403 lines
9.2 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 Runtime
|
|
import SwiftTreeSitter
|
|
import Testing
|
|
import TreeSitter
|
|
import TreeSitterP4
|
|
|
|
@testable import P4Compiler
|
|
|
|
@Test func test_simple_control_declaration() async throws {
|
|
let simple_parser_declaration = """
|
|
control simple() {
|
|
action a() {
|
|
}
|
|
table t {
|
|
key = {
|
|
true: exact;
|
|
}
|
|
}
|
|
apply {
|
|
}
|
|
};
|
|
"""
|
|
let x = { (tipe: P4Type) -> Bool in
|
|
switch tipe {
|
|
case let c as Control: c.name == "simple"
|
|
default: false
|
|
}
|
|
}
|
|
let program = try! #UseOkResult(SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration))
|
|
#expect(program.TypesWithTypes(x).count == 1)
|
|
}
|
|
|
|
@Test func test_simple_control_declaration2() async throws {
|
|
let simple_parser_declaration = """
|
|
struct Testing {
|
|
};
|
|
control simple() {
|
|
action a() {
|
|
}
|
|
table t {
|
|
key = {
|
|
true: exact;
|
|
}
|
|
}
|
|
apply {
|
|
}
|
|
};
|
|
control complex() {
|
|
action b() {
|
|
}
|
|
table t {
|
|
key = {
|
|
true: exact;
|
|
}
|
|
}
|
|
apply {
|
|
}
|
|
};
|
|
"""
|
|
|
|
let filter = { (tipe: P4Type) -> Bool in
|
|
switch tipe {
|
|
case let c as Control: c.name == "simple" || c.name == "complex"
|
|
default: false
|
|
}
|
|
}
|
|
let program = try! #UseOkResult(SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration))
|
|
#expect(program.TypesWithTypes(filter).count == 2)
|
|
}
|
|
|
|
@Test func test_simple_control_declaration_with_actions() async throws {
|
|
let simple_parser_declaration = """
|
|
control simple() {
|
|
action a() {
|
|
}
|
|
table t {
|
|
key = {
|
|
true: exact;
|
|
}
|
|
actions = {
|
|
a;
|
|
}
|
|
}
|
|
apply {
|
|
}
|
|
};
|
|
"""
|
|
let x = { (tipe: P4Type) -> Bool in
|
|
switch tipe {
|
|
case let c as Control: c.name == "simple"
|
|
default: false
|
|
}
|
|
}
|
|
let program = try! #UseOkResult(SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration))
|
|
#expect(program.TypesWithTypes(x).count == 1)
|
|
}
|
|
|
|
@Test func test_simple_control_declaration_with_misnamed_actions() async throws {
|
|
let simple_parser_declaration = """
|
|
control simple() {
|
|
action a() {
|
|
}
|
|
table t {
|
|
key = {
|
|
true: exact;
|
|
}
|
|
actions = {
|
|
b;
|
|
}
|
|
}
|
|
apply {
|
|
}
|
|
};
|
|
"""
|
|
#expect(
|
|
#RequireErrorResult(
|
|
Error(
|
|
withMessage:
|
|
"{54, 63}: Error(s) parsing property list: {91, 26}: Error(s) parsing table actions: Cannot find b in lexical scope."
|
|
),
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration))
|
|
)
|
|
}
|
|
|
|
@Test func test_simple_control_declaration_with_misnamed_actions2() async throws {
|
|
let simple_parser_declaration = """
|
|
control simple() {
|
|
action a() {
|
|
}
|
|
table t {
|
|
key = {
|
|
true: exact;
|
|
}
|
|
actions = {
|
|
a;
|
|
b;
|
|
}
|
|
}
|
|
apply {
|
|
}
|
|
};
|
|
"""
|
|
#expect(
|
|
#RequireErrorResult(
|
|
Error(
|
|
withMessage:
|
|
"{54, 72}: Error(s) parsing property list: {91, 35}: Error(s) parsing table actions: Cannot find b in lexical scope."
|
|
),
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration))
|
|
)
|
|
}
|
|
|
|
@Test func test_simple_control_declaration_with_mistyped_actions() async throws {
|
|
let simple_parser_declaration = """
|
|
bool a() {
|
|
return true;
|
|
};
|
|
control simple() {
|
|
table t {
|
|
key = {
|
|
true: exact;
|
|
}
|
|
actions = {
|
|
a;
|
|
}
|
|
}
|
|
apply {
|
|
}
|
|
};
|
|
"""
|
|
#expect(
|
|
#RequireErrorResult(
|
|
Error(
|
|
withMessage:
|
|
"{64, 63}: Error(s) parsing property list: {101, 26}: Error(s) parsing table actions: {101, 26}: a does not name an action"
|
|
),
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration))
|
|
)
|
|
}
|
|
|
|
@Test func test_simple_control_declaration_with_parameters() async throws {
|
|
let simple_parser_declaration = """
|
|
control simple(bool x, bool y) {
|
|
action a() {
|
|
}
|
|
table t {
|
|
key = {
|
|
x: exact;
|
|
y: exact;
|
|
}
|
|
}
|
|
apply {
|
|
}
|
|
};
|
|
"""
|
|
#expect(#RequireOkResult(SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration)))
|
|
}
|
|
|
|
@Test func test_simple_control_declaration_with_multiple_tables() async throws {
|
|
let simple_parser_declaration = """
|
|
control simple(bool x, bool y, bool a, bool b) {
|
|
action a() {
|
|
}
|
|
table t {
|
|
key = {
|
|
x: exact;
|
|
y: exact;
|
|
}
|
|
}
|
|
table u {
|
|
key = {
|
|
a: exact;
|
|
b: exact;
|
|
}
|
|
}
|
|
apply {
|
|
}
|
|
};
|
|
"""
|
|
#expect(
|
|
#RequireErrorResult(
|
|
Error(
|
|
withMessage: "{0, 215}: More than one table in control declaration"
|
|
),
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration))
|
|
)
|
|
}
|
|
|
|
@Test func test_simple_control_declaration_with_action_using_parameter() async throws {
|
|
let simple_parser_declaration = """
|
|
control simple(bool x, bool y) {
|
|
action a(int z) {
|
|
z = 5;
|
|
}
|
|
table t {
|
|
key = {
|
|
x: exact;
|
|
y: exact;
|
|
}
|
|
}
|
|
apply {
|
|
}
|
|
};
|
|
"""
|
|
#expect(#RequireOkResult(SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration)))
|
|
}
|
|
|
|
@Test func test_simple_control_declaration_with_action_using_parameter_wrong_type() async throws {
|
|
let simple_parser_declaration = """
|
|
control simple(bool x, bool y) {
|
|
action a(int z) {
|
|
z = false;
|
|
}
|
|
table t {
|
|
key = {
|
|
x: exact;
|
|
y: exact;
|
|
}
|
|
}
|
|
apply {
|
|
}
|
|
};
|
|
"""
|
|
#expect(
|
|
#RequireErrorResult(
|
|
Error(
|
|
withMessage:
|
|
"{57, 10}: Failed to parse a statement element: {57, 1}: Cannot assign value with type Boolean to identifier z with type Int (width: Infinite)"
|
|
),
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration))
|
|
)
|
|
}
|
|
|
|
@Test func test_simple_control_declaration_with_element_after_apply() async throws {
|
|
let simple_parser_declaration = """
|
|
control simple(bool x, bool y) {
|
|
action a(int z) {
|
|
z = false;
|
|
}
|
|
table t {
|
|
key = {
|
|
x: exact;
|
|
y: exact;
|
|
}
|
|
}
|
|
apply {
|
|
}
|
|
table x {
|
|
key = {
|
|
x: exact;
|
|
y: exact;
|
|
}
|
|
}
|
|
};
|
|
"""
|
|
#expect(
|
|
#RequireErrorResult(
|
|
Error(
|
|
withMessage:
|
|
"Could not compile the P4 program"
|
|
),
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration))
|
|
)
|
|
}
|
|
|
|
@Test func test_simple_control_declaration_with_action_with_params_right_order() async throws {
|
|
let simple_parser_declaration = """
|
|
control simple(int x, int y) {
|
|
action a(inout int ax, bool ay) {
|
|
ax = 5;
|
|
}
|
|
table t {
|
|
key = {
|
|
x: exact;
|
|
y: exact;
|
|
}
|
|
}
|
|
apply {
|
|
}
|
|
};
|
|
"""
|
|
|
|
let x = { (tipe: P4Type) -> Bool in
|
|
switch tipe {
|
|
case let c as Control: c.name == "simple"
|
|
default: false
|
|
}
|
|
}
|
|
let program = try! #UseOkResult(SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration))
|
|
#expect(program.TypesWithTypes(x).count == 1)
|
|
}
|
|
|
|
@Test func test_simple_control_declaration_with_action_with_params_wrong_order() async throws {
|
|
let simple_parser_declaration = """
|
|
control simple(int x, int y) {
|
|
action a(int ax, inout bool ay) {
|
|
ay = false;
|
|
}
|
|
table t {
|
|
key = {
|
|
x: exact;
|
|
y: exact;
|
|
}
|
|
}
|
|
apply {
|
|
}
|
|
};
|
|
"""
|
|
#expect(
|
|
#RequireErrorResult(
|
|
ErrorWithLocation(
|
|
sourceLocation: SourceLocation(41, 23),
|
|
withError: "All parameters with direction must precede directionless parameters"),
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration))
|
|
)
|
|
}
|
|
|
|
@Test func test_simple_control_declaration_with_action_with_params_wrong_order_interspersed() async throws {
|
|
let simple_parser_declaration = """
|
|
control simple(int x, int y) {
|
|
action a(inout bool aa, int ax, inout bool ay) {
|
|
aa = false;
|
|
}
|
|
table t {
|
|
key = {
|
|
x: exact;
|
|
y: exact;
|
|
}
|
|
}
|
|
apply {
|
|
}
|
|
};
|
|
"""
|
|
#expect(
|
|
#RequireErrorResult(
|
|
ErrorWithLocation(
|
|
sourceLocation: SourceLocation(41, 38),
|
|
withError: "All parameters with direction must precede directionless parameters"),
|
|
SpecialCompilers.ProgramCompiler.Compile(simple_parser_declaration))
|
|
)
|
|
}
|