61d8f601e8
In P4, parsers are considered types. Those parsers are instantiated. The instantiated parsers are values. Previously, gp4 treated a parser type and a parser value as identical. This PR makes that difference clear _and_ sets the stage for the future. TODO: Make the same distinction between control and action types and values. Signed-off-by: Will Hawkins <hawkinsw@obs.cr>
128 lines
4.1 KiB
Swift
128 lines
4.1 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
|
|
|
|
public struct ExpressionStatement {
|
|
public let expression: EvaluatableExpression
|
|
|
|
public init(_ expr: EvaluatableExpression) {
|
|
self.expression = expr
|
|
}
|
|
}
|
|
|
|
public struct Program {
|
|
public var types: [P4Type] = Array()
|
|
public var externs: [P4Type] = Array()
|
|
public var instances: [P4Value] = Array()
|
|
|
|
/// Type of closure for filtering results from ``Program/InstancesWithTypes(_:)``
|
|
public typealias TypeFilter = (P4QualifiedType) -> Bool
|
|
/// Type of closure for filtering results from ``Program/TypesWithTypes(_:)``
|
|
public typealias DataTypeFilter = (P4Type) -> Bool
|
|
|
|
/// Retrieve global instances in the compiled P4 program.
|
|
public func InstancesWithTypes() -> [P4Value] {
|
|
return self.instances
|
|
}
|
|
|
|
/// Retrieve global instances in the compiled P4 program.
|
|
///
|
|
/// Use the given filter to select which of the global instances
|
|
/// from the compiled P4 program to retrieve.
|
|
///
|
|
/// If the compiled P4 program (from the source in the
|
|
/// string `p4_program_with_control_decl`) has two Control
|
|
/// instances and you only want to select the one named simple,
|
|
/// you could use a filter like
|
|
///
|
|
/// @Snippet(path: "use-program-instanceswithtypes", slice: "include")
|
|
///
|
|
public func InstancesWithTypes(_ filter: TypeFilter) -> [P4Value] {
|
|
return self.instances.filter { instance in
|
|
filter(instance.type())
|
|
}
|
|
}
|
|
|
|
/// Retrieve global types in the compiled P4 program.
|
|
public func TypesWithTypes() -> [P4Type] {
|
|
return self.types
|
|
}
|
|
|
|
/// Retrieve global types declared in the compiled P4 program.
|
|
///
|
|
/// Use the given filter to select which of the global types
|
|
/// declared in the compiled P4 program to retrieve.
|
|
///
|
|
/// If the compiled P4 program (from the source in the
|
|
/// string `p4_program_with_struct_decl`) has two structs declared and
|
|
/// you only want to select the one named `agg`, you could
|
|
/// use a filter like
|
|
///
|
|
/// @Snippet(path: "use-program-typeswithtypes", slice: "include")
|
|
///
|
|
public func TypesWithTypes(_ filter: DataTypeFilter) -> [P4Type] {
|
|
return self.types.filter { instance in
|
|
filter(instance)
|
|
}
|
|
}
|
|
|
|
/// Retrieve extern types in the compiled P4 program.
|
|
public func Externs() -> [P4Type] {
|
|
return self.externs
|
|
}
|
|
|
|
/// Retrieve extern types declared in the compiled P4 program.
|
|
///
|
|
/// Use the given filter to select which of the extern types
|
|
/// declared in the compiled P4 program to retrieve.
|
|
///
|
|
/// If the compiled P4 program (from the source in the
|
|
/// string `p4_program_with_struct_decl`) has two extern structs declared and
|
|
/// you only want to select the one named `agg`, you could
|
|
/// use a filter like
|
|
///
|
|
/// @Snippet(path: "use-program-typeswithtypes", slice: "include")
|
|
///
|
|
public func Externs(_ filter: DataTypeFilter) -> [P4Type] {
|
|
return self.externs.filter { instance in
|
|
filter(instance)
|
|
}
|
|
}
|
|
|
|
/// Find the program's main parser
|
|
///
|
|
/// Note: For now, the main parser is expected to be named main_parser.
|
|
public func starting_parser() -> Result<Parser> {
|
|
return self.find_parser(withName: Identifier(name: "main_parser"))
|
|
}
|
|
|
|
public func find_parser(withName name: Identifier) -> Result<Parser> {
|
|
for instance in self.types {
|
|
guard let parser = instance as? Parser else {
|
|
continue
|
|
}
|
|
if parser.name == name {
|
|
return .Ok(parser)
|
|
}
|
|
}
|
|
return .Error(Error(withMessage: "Could not find parser named \(name)"))
|
|
}
|
|
|
|
public init() {}
|
|
}
|