35b2537754
Add support for directions on parameters and attributed types. The latter is necessary to support the former, but is not limited to the direction use case. An attributed type is a P4 type with attributes (like its direction, whether it is read only, etc.) Other attributes will be added in the future. Changes necessary to support attributed types include: 1. Global instances tracked during compilation are not attributed types and not simply types. 2. (future) Type checking will have to make sure that a types attributes do not affect type compatibility. Signed-off-by: Will Hawkins <hawkinsw@obs.cr>
135 lines
3.9 KiB
Swift
135 lines
3.9 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 Parameter: CustomStringConvertible, Equatable {
|
|
public static func == (lhs: Parameter, rhs: Parameter) -> Bool {
|
|
return lhs.name == rhs.name && lhs.type.eq(rhs: rhs.type) && lhs.direction == rhs.direction
|
|
}
|
|
|
|
public var name: Identifier
|
|
public var type: P4Type
|
|
public var direction: Direction?
|
|
|
|
public init(
|
|
identifier: Identifier, withType type: P4Type, withDirection direction: Direction? = .none
|
|
) {
|
|
self.name = identifier
|
|
self.type = type
|
|
self.direction = direction
|
|
}
|
|
|
|
public var description: String {
|
|
let direction = self.direction != .none ? self.direction!.description : "no"
|
|
return "Parameter: \(self.name) with type \(self.type) with \(direction) direction"
|
|
}
|
|
|
|
/// Calculate whether the `argument` is compatible with this parameter.
|
|
public func compatible(_ argument: Argument) -> Bool {
|
|
let arg_type = argument.argument.type()
|
|
return arg_type.eq(rhs: self.type)
|
|
}
|
|
|
|
public func attributedType() -> P4TypeAttributed {
|
|
return P4TypeAttributed(
|
|
self.type, self.direction == .none ? [] : [P4TypeAttribute.Direction(self.direction!)])
|
|
}
|
|
}
|
|
|
|
public struct ParameterList: CustomStringConvertible, Equatable {
|
|
public static func == (lhs: ParameterList, rhs: ParameterList) -> Bool {
|
|
if lhs.parameters.count != rhs.parameters.count {
|
|
return false
|
|
}
|
|
|
|
return 0
|
|
== zip(lhs.parameters, rhs.parameters).count { (lparam, rparam) in
|
|
return lparam != rparam
|
|
}
|
|
}
|
|
|
|
public var parameters: [Parameter]
|
|
|
|
public init() {
|
|
self.parameters = Array()
|
|
}
|
|
|
|
public init(_ parameters: [Parameter]) {
|
|
self.parameters = parameters
|
|
}
|
|
|
|
public func addParameter(_ parameter: Parameter) -> ParameterList {
|
|
return ParameterList(self.parameters + [parameter])
|
|
}
|
|
|
|
public var description: String {
|
|
let parameters = self.parameters.map { parameter in
|
|
parameter.description
|
|
}.joined(separator: ";")
|
|
return "Parameter list: \(parameters)"
|
|
}
|
|
}
|
|
|
|
public struct ArgumentList {
|
|
public let arguments: [Argument]
|
|
|
|
public init(_ arguments: [Argument]) {
|
|
self.arguments = arguments
|
|
}
|
|
|
|
public func compatible(_ parameters: ParameterList) -> Result<()> {
|
|
if self.arguments.count != parameters.parameters.count {
|
|
return .Error(
|
|
Error(
|
|
withMessage:
|
|
"\(self.arguments.count) arguments found but \(parameters.parameters.count) required"))
|
|
}
|
|
|
|
for (arg, param) in zip(self.arguments, parameters.parameters) {
|
|
let arg_index = arg.index
|
|
let arg_type = arg.argument.type()
|
|
if !param.compatible(arg) {
|
|
return .Error(
|
|
Error(
|
|
withMessage:
|
|
"Argument \(arg_index)'s type (\(arg_type)) is incompatible with the parameter type (\(param.type))"
|
|
))
|
|
}
|
|
}
|
|
return .Ok(())
|
|
}
|
|
|
|
public func addArgument(_ argument: Argument) -> ArgumentList {
|
|
return ArgumentList(self.arguments + [argument])
|
|
}
|
|
|
|
public func count() -> Int {
|
|
return self.arguments.count
|
|
}
|
|
}
|
|
|
|
public struct Argument {
|
|
public let index: Int
|
|
public let argument: EvaluatableExpression
|
|
|
|
public init(_ argument: EvaluatableExpression, atIndex index: Int) {
|
|
self.argument = argument
|
|
self.index = index
|
|
}
|
|
}
|