use crate::emitter::{identifier_with_prefix_suffix, symbol_string, SymbolContext};
use std::collections::HashMap;
use veryl_analyzer::symbol::{GenericMap, SymbolKind};
use veryl_analyzer::symbol_table;
use veryl_metadata::{Build, BuiltinType, Metadata};
use veryl_parser::resource_table::StrId;
use veryl_parser::veryl_grammar_trait::*;
use veryl_parser::veryl_token::{Token, VerylToken};
use veryl_parser::veryl_walker::VerylWalker;
use veryl_parser::Stringifier;
#[derive(Debug, Default, Clone, Copy, Eq, PartialEq, Hash)]
pub struct Location {
pub line: u32,
pub column: u32,
pub length: u32,
pub duplicated: Option<usize>,
}
impl From<&Token> for Location {
fn from(x: &Token) -> Self {
Self {
line: x.line,
column: x.column,
length: x.length,
duplicated: None,
}
}
}
impl From<Token> for Location {
fn from(x: Token) -> Self {
Self {
line: x.line,
column: x.column,
length: x.length,
duplicated: None,
}
}
}
#[derive(Default)]
pub struct Align {
enable: bool,
index: usize,
max_width: u32,
width: u32,
line: u32,
rest: Vec<(Location, u32)>,
additions: HashMap<Location, u32>,
last_location: Option<Location>,
}
impl Align {
fn finish_group(&mut self) {
for (loc, width) in &self.rest {
self.additions.insert(*loc, self.max_width - width);
}
self.rest.clear();
self.max_width = 0;
}
fn finish_item(&mut self) {
assert!(self.enable);
self.enable = false;
if let Some(loc) = self.last_location {
if loc.line - self.line > 1 {
self.finish_group();
}
self.max_width = u32::max(self.max_width, self.width);
self.line = loc.line;
self.rest.push((loc, self.width));
self.width = 0;
self.index += 1;
}
}
fn start_item(&mut self) {
assert!(!self.enable);
self.enable = true;
self.width = 0;
}
fn token(&mut self, x: &VerylToken) {
if self.enable {
self.width += x.token.length;
let loc: Location = x.token.into();
self.last_location = Some(loc);
}
}
fn dummy_location(&mut self, x: Location) {
if self.enable {
self.width += 0; self.last_location = Some(x);
}
}
fn duplicated_token(&mut self, x: &VerylToken, i: usize) {
if self.enable {
self.width += x.token.length;
let mut loc: Location = x.token.into();
loc.duplicated = Some(i);
self.last_location = Some(loc);
}
}
fn space(&mut self, x: usize) {
if self.enable {
self.width += x as u32;
}
}
}
mod align_kind {
pub const IDENTIFIER: usize = 0;
pub const TYPE: usize = 1;
pub const EXPRESSION: usize = 2;
pub const WIDTH: usize = 3;
pub const ARRAY: usize = 4;
pub const ASSIGNMENT: usize = 5;
pub const PARAMETER: usize = 6;
pub const DIRECTION: usize = 7;
}
#[derive(Default)]
pub struct Aligner {
pub additions: HashMap<Location, u32>,
aligns: [Align; 8],
in_type_expression: bool,
in_import: bool,
project_name: Option<StrId>,
build_opt: Build,
generic_map: Vec<GenericMap>,
}
impl Aligner {
pub fn new() -> Self {
Default::default()
}
pub fn set_metadata(&mut self, metadata: &Metadata) {
self.project_name = Some(metadata.project.name.as_str().into());
self.build_opt = metadata.build.clone();
}
pub fn align(&mut self, input: &Veryl) {
self.veryl(input);
self.finish_group();
for align in &self.aligns {
for (x, y) in &align.additions {
self.additions
.entry(*x)
.and_modify(|val| *val += *y)
.or_insert(*y);
}
}
}
fn finish_group(&mut self) {
for i in 0..self.aligns.len() {
self.aligns[i].finish_group();
}
}
fn reset_align(&mut self) {
self.finish_group();
}
fn space(&mut self, repeat: usize) {
for i in 0..self.aligns.len() {
self.aligns[i].space(repeat);
}
}
fn is_implicit_scalar_type(&mut self, x: &ScalarType) -> bool {
let mut stringifier = Stringifier::new();
stringifier.scalar_type(x);
let r#type = match stringifier.as_str() {
"u32" => Some(BuiltinType::U32),
"u64" => Some(BuiltinType::U64),
"i32" => Some(BuiltinType::I32),
"i64" => Some(BuiltinType::I64),
"f32" => Some(BuiltinType::F32),
"f64" => Some(BuiltinType::F64),
"string" => Some(BuiltinType::String),
_ => None,
};
if let Some(x) = r#type {
self.build_opt.implicit_parameter_types.contains(&x)
} else {
false
}
}
fn is_implicit_type(&mut self) -> bool {
self.build_opt
.implicit_parameter_types
.contains(&BuiltinType::Type)
}
}
impl VerylWalker for Aligner {
fn veryl_token(&mut self, arg: &VerylToken) {
for i in 0..self.aligns.len() {
self.aligns[i].token(arg);
}
}
fn clock(&mut self, arg: &Clock) {
self.veryl_token(&arg.clock_token.replace("logic"));
}
fn clock_posedge(&mut self, arg: &ClockPosedge) {
self.veryl_token(&arg.clock_posedge_token.replace("logic"));
}
fn clock_negedge(&mut self, arg: &ClockNegedge) {
self.veryl_token(&arg.clock_negedge_token.replace("logic"));
}
fn r#const(&mut self, arg: &Const) {
self.veryl_token(&arg.const_token.replace("localparam"));
}
fn reset(&mut self, arg: &Reset) {
self.veryl_token(&arg.reset_token.replace("logic"));
}
fn reset_async_high(&mut self, arg: &ResetAsyncHigh) {
self.veryl_token(&arg.reset_async_high_token.replace("logic"));
}
fn reset_async_low(&mut self, arg: &ResetAsyncLow) {
self.veryl_token(&arg.reset_async_low_token.replace("logic"));
}
fn reset_sync_high(&mut self, arg: &ResetSyncHigh) {
self.veryl_token(&arg.reset_sync_high_token.replace("logic"));
}
fn reset_sync_low(&mut self, arg: &ResetSyncLow) {
self.veryl_token(&arg.reset_sync_low_token.replace("logic"));
}
fn f32(&mut self, arg: &F32) {
self.veryl_token(&arg.f32_token.replace("shortreal"));
}
fn f64(&mut self, arg: &F64) {
self.veryl_token(&arg.f64_token.replace("real"));
}
fn i32(&mut self, arg: &I32) {
self.veryl_token(&arg.i32_token.replace("int signed"));
}
fn i64(&mut self, arg: &I64) {
self.veryl_token(&arg.i64_token.replace("longint signed"));
}
fn param(&mut self, arg: &Param) {
self.veryl_token(&arg.param_token.replace("parameter"));
}
fn u32(&mut self, arg: &U32) {
self.veryl_token(&arg.u32_token.replace("int unsigned"));
}
fn u64(&mut self, arg: &U64) {
self.veryl_token(&arg.u64_token.replace("longint unsigned"));
}
fn identifier(&mut self, arg: &Identifier) {
let (prefix, suffix) = if let Ok(found) = symbol_table::resolve(arg) {
match &found.found.kind {
SymbolKind::Port(x) => (x.prefix.clone(), x.suffix.clone()),
SymbolKind::Variable(x) => (x.prefix.clone(), x.suffix.clone()),
_ => (None, None),
}
} else {
(None, None)
};
self.veryl_token(&identifier_with_prefix_suffix(arg, &prefix, &suffix));
}
fn hierarchical_identifier(&mut self, arg: &HierarchicalIdentifier) {
let list_len = &arg.hierarchical_identifier_list0.len();
let (prefix, suffix) = if let Ok(found) = symbol_table::resolve(arg) {
match &found.found.kind {
SymbolKind::Port(x) => (x.prefix.clone(), x.suffix.clone()),
SymbolKind::Variable(x) => (x.prefix.clone(), x.suffix.clone()),
_ => (None, None),
}
} else {
unreachable!()
};
if *list_len == 0 {
self.veryl_token(&identifier_with_prefix_suffix(
&arg.identifier,
&prefix,
&suffix,
));
} else {
self.identifier(&arg.identifier);
}
for x in &arg.hierarchical_identifier_list {
self.select(&x.select);
}
for (i, x) in arg.hierarchical_identifier_list0.iter().enumerate() {
self.dot(&x.dot);
if (i + 1) == *list_len {
self.veryl_token(&identifier_with_prefix_suffix(
&x.identifier,
&prefix,
&suffix,
));
} else {
self.identifier(&x.identifier);
}
for x in &x.hierarchical_identifier_list0_list {
self.select(&x.select);
}
}
}
fn scoped_identifier(&mut self, arg: &ScopedIdentifier) {
if let Ok(symbol) = symbol_table::resolve(arg) {
let context: SymbolContext = self.into();
let text = symbol_string(arg.identifier(), &symbol.found, &context);
self.veryl_token(&arg.identifier().replace(&text));
}
}
fn expression(&mut self, arg: &Expression) {
self.expression01(&arg.expression01);
for x in &arg.expression_list {
self.space(1);
self.operator01(&x.operator01);
self.space(1);
self.expression01(&x.expression01);
}
}
fn expression01(&mut self, arg: &Expression01) {
self.expression02(&arg.expression02);
for x in &arg.expression01_list {
self.space(1);
self.operator02(&x.operator02);
self.space(1);
self.expression02(&x.expression02);
}
}
fn expression02(&mut self, arg: &Expression02) {
self.expression03(&arg.expression03);
for x in &arg.expression02_list {
self.space(1);
self.operator03(&x.operator03);
self.space(1);
self.expression03(&x.expression03);
}
}
fn expression03(&mut self, arg: &Expression03) {
self.expression04(&arg.expression04);
for x in &arg.expression03_list {
self.space(1);
self.operator04(&x.operator04);
self.space(1);
self.expression04(&x.expression04);
}
}
fn expression04(&mut self, arg: &Expression04) {
self.expression05(&arg.expression05);
for x in &arg.expression04_list {
self.space(1);
self.operator05(&x.operator05);
self.space(1);
self.expression05(&x.expression05);
}
}
fn expression05(&mut self, arg: &Expression05) {
self.expression06(&arg.expression06);
for x in &arg.expression05_list {
self.space(1);
self.operator06(&x.operator06);
self.space(1);
self.expression06(&x.expression06);
}
}
fn expression06(&mut self, arg: &Expression06) {
self.expression07(&arg.expression07);
for x in &arg.expression06_list {
self.space(1);
self.operator07(&x.operator07);
self.space(1);
self.expression07(&x.expression07);
}
}
fn expression07(&mut self, arg: &Expression07) {
self.expression08(&arg.expression08);
for x in &arg.expression07_list {
self.space(1);
self.operator08(&x.operator08);
self.space(1);
self.expression08(&x.expression08);
}
}
fn expression08(&mut self, arg: &Expression08) {
self.expression09(&arg.expression09);
for x in &arg.expression08_list {
self.space(1);
self.operator09(&x.operator09);
self.space(1);
self.expression09(&x.expression09);
}
}
fn expression09(&mut self, arg: &Expression09) {
self.expression10(&arg.expression10);
for x in &arg.expression09_list {
self.space(1);
match &*x.expression09_list_group {
Expression09ListGroup::Operator10(x) => self.operator10(&x.operator10),
Expression09ListGroup::Star(x) => self.star(&x.star),
}
self.space(1);
self.expression10(&x.expression10);
}
}
fn expression10(&mut self, arg: &Expression10) {
self.expression11(&arg.expression11);
for x in &arg.expression10_list {
self.space(1);
self.operator11(&x.operator11);
self.space(1);
self.expression11(&x.expression11);
}
}
fn argument_list(&mut self, arg: &ArgumentList) {
self.argument_item(&arg.argument_item);
for x in &arg.argument_list_list {
self.comma(&x.comma);
self.space(1);
self.argument_item(&x.argument_item);
}
if let Some(ref x) = arg.argument_list_opt {
self.comma(&x.comma);
}
}
fn type_expression(&mut self, arg: &TypeExpression) {
self.in_type_expression = true;
match arg {
TypeExpression::ScalarType(x) => self.scalar_type(&x.scalar_type),
TypeExpression::TypeLParenExpressionRParen(x) => {
self.r#type(&x.r#type);
self.l_paren(&x.l_paren);
self.expression(&x.expression);
self.r_paren(&x.r_paren);
}
}
self.in_type_expression = false;
}
fn width(&mut self, arg: &Width) {
self.l_angle(&arg.l_angle);
self.expression(&arg.expression);
self.space("-1:0".len());
for x in &arg.width_list {
self.space("][".len());
self.expression(&x.expression);
self.space("-1:0".len());
}
self.r_angle(&arg.r_angle);
}
fn array(&mut self, arg: &Array) {
self.l_bracket(&arg.l_bracket);
self.expression(&arg.expression);
self.space("-1:0".len());
for x in &arg.array_list {
self.space("][".len());
self.expression(&x.expression);
self.space("-1:0".len());
}
self.r_bracket(&arg.r_bracket);
}
fn scalar_type(&mut self, arg: &ScalarType) {
if !self.in_type_expression {
self.aligns[align_kind::TYPE].start_item();
self.space(1);
}
for x in &arg.scalar_type_list {
self.type_modifier(&x.type_modifier);
self.space(1);
}
match &*arg.scalar_type_group {
ScalarTypeGroup::VariableTypeScalarTypeOpt(x) => {
self.variable_type(&x.variable_type);
if !self.in_type_expression {
self.aligns[align_kind::TYPE].finish_item();
self.aligns[align_kind::WIDTH].start_item();
}
if let Some(ref x) = x.scalar_type_opt {
self.space(1);
self.width(&x.width);
} else if !self.in_type_expression {
let loc = self.aligns[align_kind::TYPE].last_location;
self.aligns[align_kind::WIDTH].dummy_location(loc.unwrap());
}
}
ScalarTypeGroup::FixedType(x) => {
self.fixed_type(&x.fixed_type);
if !self.in_type_expression {
self.aligns[align_kind::TYPE].finish_item();
self.aligns[align_kind::WIDTH].start_item();
let loc = self.aligns[align_kind::TYPE].last_location;
self.aligns[align_kind::WIDTH].dummy_location(loc.unwrap());
}
}
}
if !self.in_type_expression {
self.aligns[align_kind::WIDTH].finish_item();
}
}
fn array_type(&mut self, arg: &ArrayType) {
self.scalar_type(&arg.scalar_type);
self.aligns[align_kind::ARRAY].start_item();
if let Some(ref x) = arg.array_type_opt {
self.space(1);
self.array(&x.array);
} else {
let loc = self.aligns[align_kind::IDENTIFIER].last_location;
self.aligns[align_kind::ARRAY].dummy_location(loc.unwrap());
}
self.aligns[align_kind::ARRAY].finish_item();
}
fn let_statement(&mut self, arg: &LetStatement) {
self.r#let(&arg.r#let);
self.aligns[align_kind::IDENTIFIER].start_item();
self.identifier(&arg.identifier);
self.aligns[align_kind::IDENTIFIER].finish_item();
self.colon(&arg.colon);
self.array_type(&arg.array_type);
self.equ(&arg.equ);
self.expression(&arg.expression);
self.semicolon(&arg.semicolon);
}
fn identifier_statement(&mut self, arg: &IdentifierStatement) {
self.aligns[align_kind::IDENTIFIER].start_item();
self.expression_identifier(&arg.expression_identifier);
self.aligns[align_kind::IDENTIFIER].finish_item();
match &*arg.identifier_statement_group {
IdentifierStatementGroup::FunctionCall(x) => {
self.function_call(&x.function_call);
}
IdentifierStatementGroup::Assignment(x) => {
self.assignment(&x.assignment);
}
}
self.semicolon(&arg.semicolon);
}
fn assignment(&mut self, arg: &Assignment) {
self.aligns[align_kind::ASSIGNMENT].start_item();
match &*arg.assignment_group {
AssignmentGroup::Equ(x) => self.equ(&x.equ),
AssignmentGroup::AssignmentOperator(x) => {
self.assignment_operator(&x.assignment_operator)
}
}
self.aligns[align_kind::ASSIGNMENT].finish_item();
self.expression(&arg.expression);
}
fn case_item(&mut self, arg: &CaseItem) {
self.aligns[align_kind::EXPRESSION].start_item();
match &*arg.case_item_group {
CaseItemGroup::CaseCondition(x) => {
self.range_item(&x.case_condition.range_item);
for x in &x.case_condition.case_condition_list {
self.comma(&x.comma);
self.space(1);
self.range_item(&x.range_item);
}
}
CaseItemGroup::Defaul(x) => self.defaul(&x.defaul),
}
self.aligns[align_kind::EXPRESSION].finish_item();
self.colon(&arg.colon);
match &*arg.case_item_group0 {
CaseItemGroup0::Statement(x) => self.statement(&x.statement),
CaseItemGroup0::StatementBlock(x) => self.statement_block(&x.statement_block),
}
}
fn switch_item(&mut self, arg: &SwitchItem) {
self.aligns[align_kind::EXPRESSION].start_item();
match &*arg.switch_item_group {
SwitchItemGroup::SwitchCondition(x) => {
self.expression(&x.switch_condition.expression);
for x in &x.switch_condition.switch_condition_list {
self.comma(&x.comma);
self.space(1);
self.expression(&x.expression);
}
}
SwitchItemGroup::Defaul(x) => self.defaul(&x.defaul),
}
self.aligns[align_kind::EXPRESSION].finish_item();
self.colon(&arg.colon);
match &*arg.switch_item_group0 {
SwitchItemGroup0::Statement(x) => self.statement(&x.statement),
SwitchItemGroup0::StatementBlock(x) => self.statement_block(&x.statement_block),
}
}
fn let_declaration(&mut self, arg: &LetDeclaration) {
self.r#let(&arg.r#let);
self.aligns[align_kind::IDENTIFIER].start_item();
self.identifier(&arg.identifier);
self.aligns[align_kind::IDENTIFIER].finish_item();
self.colon(&arg.colon);
self.array_type(&arg.array_type);
self.equ(&arg.equ);
self.expression(&arg.expression);
self.semicolon(&arg.semicolon);
}
fn var_declaration(&mut self, arg: &VarDeclaration) {
self.var(&arg.var);
self.aligns[align_kind::IDENTIFIER].start_item();
self.identifier(&arg.identifier);
self.aligns[align_kind::IDENTIFIER].finish_item();
self.colon(&arg.colon);
self.array_type(&arg.array_type);
self.semicolon(&arg.semicolon);
}
fn const_declaration(&mut self, arg: &ConstDeclaration) {
self.r#const(&arg.r#const);
self.aligns[align_kind::IDENTIFIER].start_item();
self.identifier(&arg.identifier);
self.aligns[align_kind::IDENTIFIER].finish_item();
self.colon(&arg.colon);
match &*arg.const_declaration_group {
ConstDeclarationGroup::ArrayTypeEquExpression(x) => {
if !self.is_implicit_scalar_type(&x.array_type.scalar_type) {
self.array_type(&x.array_type);
} else {
self.aligns[align_kind::TYPE].start_item();
self.aligns[align_kind::TYPE]
.dummy_location(arg.r#const.const_token.token.into());
self.aligns[align_kind::TYPE].finish_item();
}
self.equ(&x.equ);
self.expression(&x.expression);
}
ConstDeclarationGroup::TypeEquTypeExpression(x) => {
self.aligns[align_kind::TYPE].start_item();
if !self.is_implicit_type() {
self.r#type(&x.r#type);
self.space(1);
} else {
self.aligns[align_kind::TYPE]
.dummy_location(arg.r#const.const_token.token.into());
}
self.aligns[align_kind::TYPE].finish_item();
self.equ(&x.equ);
self.type_expression(&x.type_expression);
}
}
self.semicolon(&arg.semicolon);
}
fn type_def_declaration(&mut self, arg: &TypeDefDeclaration) {
self.r#type(&arg.r#type);
self.scalar_type(&arg.array_type.scalar_type);
self.aligns[align_kind::IDENTIFIER].start_item();
self.identifier(&arg.identifier);
self.aligns[align_kind::IDENTIFIER].finish_item();
if let Some(ato) = &arg.array_type.array_type_opt {
self.aligns[align_kind::ARRAY].start_item();
self.array(&ato.array);
self.aligns[align_kind::ARRAY].finish_item();
}
self.semicolon(&arg.semicolon);
}
fn assign_declaration(&mut self, arg: &AssignDeclaration) {
self.assign(&arg.assign);
self.aligns[align_kind::IDENTIFIER].start_item();
self.hierarchical_identifier(&arg.hierarchical_identifier);
self.aligns[align_kind::IDENTIFIER].finish_item();
self.equ(&arg.equ);
self.expression(&arg.expression);
self.semicolon(&arg.semicolon);
}
fn modport_item(&mut self, arg: &ModportItem) {
self.aligns[align_kind::IDENTIFIER].start_item();
self.identifier(&arg.identifier);
self.aligns[align_kind::IDENTIFIER].finish_item();
self.colon(&arg.colon);
self.direction(&arg.direction);
}
fn struct_union_item(&mut self, arg: &StructUnionItem) {
self.aligns[align_kind::IDENTIFIER].start_item();
self.identifier(&arg.identifier);
self.aligns[align_kind::IDENTIFIER].finish_item();
self.colon(&arg.colon);
self.scalar_type(&arg.scalar_type);
}
fn inst_declaration(&mut self, arg: &InstDeclaration) {
let single_line = arg.inst_declaration_opt1.is_none();
self.inst(&arg.inst);
if single_line {
self.aligns[align_kind::IDENTIFIER].start_item();
}
self.identifier(&arg.identifier);
if single_line {
self.aligns[align_kind::IDENTIFIER].finish_item();
}
self.colon(&arg.colon);
self.scoped_identifier(&arg.scoped_identifier);
if single_line {
return;
}
if let Some(ref x) = arg.inst_declaration_opt {
self.array(&x.array);
}
if let Some(ref x) = arg.inst_declaration_opt0 {
self.inst_parameter(&x.inst_parameter);
}
if let Some(ref x) = arg.inst_declaration_opt1 {
self.l_paren(&x.l_paren);
if let Some(ref x) = x.inst_declaration_opt2 {
self.inst_port_list(&x.inst_port_list);
}
self.r_paren(&x.r_paren);
}
self.semicolon(&arg.semicolon);
}
fn inst_parameter_item(&mut self, arg: &InstParameterItem) {
self.aligns[align_kind::IDENTIFIER].start_item();
self.identifier(&arg.identifier);
self.aligns[align_kind::IDENTIFIER].finish_item();
if let Some(ref x) = arg.inst_parameter_item_opt {
self.colon(&x.colon);
self.space(1);
self.aligns[align_kind::EXPRESSION].start_item();
self.expression(&x.expression);
self.aligns[align_kind::EXPRESSION].finish_item();
} else {
self.aligns[align_kind::EXPRESSION].start_item();
self.aligns[align_kind::EXPRESSION]
.duplicated_token(&arg.identifier.identifier_token, 0);
self.aligns[align_kind::EXPRESSION].finish_item();
}
}
fn inst_port_item(&mut self, arg: &InstPortItem) {
self.aligns[align_kind::IDENTIFIER].start_item();
self.identifier(&arg.identifier);
self.aligns[align_kind::IDENTIFIER].finish_item();
if let Some(ref x) = arg.inst_port_item_opt {
self.colon(&x.colon);
self.space(1);
self.aligns[align_kind::EXPRESSION].start_item();
self.expression(&x.expression);
self.aligns[align_kind::EXPRESSION].finish_item();
} else {
let (prefix, suffix) = if let Ok(found) = symbol_table::resolve(arg.identifier.as_ref())
{
match &found.found.kind {
SymbolKind::Port(x) => (x.prefix.clone(), x.suffix.clone()),
SymbolKind::Variable(x) => (x.prefix.clone(), x.suffix.clone()),
_ => (None, None),
}
} else {
unreachable!()
};
let token = identifier_with_prefix_suffix(&arg.identifier, &prefix, &suffix);
self.aligns[align_kind::EXPRESSION].start_item();
self.aligns[align_kind::EXPRESSION].duplicated_token(&token, 0);
self.aligns[align_kind::EXPRESSION].finish_item();
}
}
fn with_parameter_item(&mut self, arg: &WithParameterItem) {
self.aligns[align_kind::PARAMETER].start_item();
match &*arg.with_parameter_item_group {
WithParameterItemGroup::Param(x) => self.param(&x.param),
WithParameterItemGroup::Const(x) => self.r#const(&x.r#const),
};
self.aligns[align_kind::PARAMETER].finish_item();
self.aligns[align_kind::IDENTIFIER].start_item();
self.identifier(&arg.identifier);
self.aligns[align_kind::IDENTIFIER].finish_item();
self.colon(&arg.colon);
match &*arg.with_parameter_item_group0 {
WithParameterItemGroup0::ArrayTypeEquExpression(x) => {
if !self.is_implicit_scalar_type(&x.array_type.scalar_type) {
self.array_type(&x.array_type);
} else {
self.aligns[align_kind::TYPE].start_item();
let loc = self.aligns[align_kind::PARAMETER].last_location;
self.aligns[align_kind::TYPE].dummy_location(loc.unwrap());
self.aligns[align_kind::TYPE].finish_item();
}
self.equ(&x.equ);
self.aligns[align_kind::EXPRESSION].start_item();
self.expression(&x.expression);
self.aligns[align_kind::EXPRESSION].finish_item();
}
WithParameterItemGroup0::TypeEquTypeExpression(x) => {
self.aligns[align_kind::TYPE].start_item();
if !self.is_implicit_type() {
self.r#type(&x.r#type);
self.space(1);
} else {
let loc = self.aligns[align_kind::PARAMETER].last_location;
self.aligns[align_kind::TYPE].dummy_location(loc.unwrap());
}
self.aligns[align_kind::TYPE].finish_item();
self.equ(&x.equ);
self.aligns[align_kind::EXPRESSION].start_item();
self.type_expression(&x.type_expression);
self.aligns[align_kind::EXPRESSION].finish_item();
}
}
}
fn port_declaration_item(&mut self, arg: &PortDeclarationItem) {
self.aligns[align_kind::IDENTIFIER].start_item();
self.identifier(&arg.identifier);
self.aligns[align_kind::IDENTIFIER].finish_item();
self.colon(&arg.colon);
match &*arg.port_declaration_item_group {
PortDeclarationItemGroup::PortTypeConcrete(x) => {
let x = x.port_type_concrete.as_ref();
self.direction(&x.direction);
self.array_type(&x.array_type);
}
PortDeclarationItemGroup::PortTypeAbstract(x) => {
let x = x.port_type_abstract.as_ref();
self.interface(&x.interface);
if let Some(ref x) = x.port_type_abstract_opt0 {
self.array(&x.array);
}
}
}
}
fn direction(&mut self, arg: &Direction) {
if !matches!(arg, Direction::Modport(_)) {
self.aligns[align_kind::DIRECTION].start_item();
}
match arg {
Direction::Input(x) => self.input(&x.input),
Direction::Output(x) => self.output(&x.output),
Direction::Inout(x) => self.inout(&x.inout),
Direction::Ref(x) => self.r#ref(&x.r#ref),
Direction::Modport(_) => (),
Direction::Import(x) => self.import(&x.import),
};
if !matches!(arg, Direction::Modport(_)) {
self.aligns[align_kind::DIRECTION].finish_item();
}
}
fn function_declaration(&mut self, arg: &FunctionDeclaration) {
self.function(&arg.function);
self.identifier(&arg.identifier);
if let Some(ref x) = arg.function_declaration_opt0 {
self.port_declaration(&x.port_declaration);
}
if let Some(ref x) = arg.function_declaration_opt1 {
self.minus_g_t(&x.minus_g_t);
self.scalar_type(&x.scalar_type);
self.reset_align();
}
self.statement_block(&arg.statement_block);
}
fn import_declaration(&mut self, arg: &ImportDeclaration) {
self.in_import = true;
self.import(&arg.import);
self.scoped_identifier(&arg.scoped_identifier);
if let Some(ref x) = arg.import_declaration_opt {
self.colon_colon(&x.colon_colon);
self.star(&x.star);
}
self.semicolon(&arg.semicolon);
self.in_import = false;
}
}
impl From<&mut Aligner> for SymbolContext {
fn from(value: &mut Aligner) -> Self {
SymbolContext {
project_name: value.project_name,
build_opt: value.build_opt.clone(),
in_import: value.in_import,
generic_map: value.generic_map.clone(),
}
}
}