// 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 . import Common import P4Lang /// The runtime for a parser public class ParserRuntime: CustomStringConvertible { public var parser: ParserInstance init(execution: ParserInstance) { self.parser = execution } /// Create a parser runtime from a P4 program public static func create(program: P4Lang.Program) -> Result { return switch program.starting_parser() { case .Ok(let parser): switch ParserInstance.compile(parser) { case .Ok(let execution): .Ok(P4Runtime.ParserRuntime(execution: execution)) case .Error(let error): .Error(error) } case .Error(let error): .Error(error) } } /// Run the P4 parser on a given packet public func run() -> Result<(ParserState, ProgramExecution)> { let (end_state, execution) = parser.execute() if let error = execution.getError() { return .Error(error) } return .Ok((end_state, execution)) } public var description: String { return "Runtime:\nExecution: \(parser)" } } /// Instances of parsers are executable extension ParserInstance: ParserExecution { public func execute() -> (ParserState, ProgramExecution) { var execution = self as ProgramExecution execution = execution.enter_scope() // First, add every state to the scope! for state in self.states { execution = execution.declare(identifier: state.state().state, withValue: state) } var c = self.start_state // Evaluate until the state is either accept or reject. while !c.done() && !execution.hasError() { (c, execution) = c.execute(program: execution) } return (c.state(), execution) } }