1#[doc(hidden)]
11pub use crate::helpers::{CodeLocation, OptionalCodeLocation};
12
13#[cfg(feature = "pt-serde")]
14use serde::{Deserialize, Serialize};
15
16#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)]
18#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
19pub enum Loc {
20 Builtin,
22 CommandLine,
24 Implicit,
26 Codegen,
28 File(usize, usize, usize),
30}
31
32impl Default for Loc {
33 fn default() -> Self {
34 Self::File(0, 0, 0)
35 }
36}
37
38#[inline(never)]
39#[cold]
40#[track_caller]
41fn not_a_file() -> ! {
42 panic!("location is not a file")
43}
44
45impl Loc {
46 #[inline]
48 pub fn begin_range(&self) -> Self {
49 match self {
50 Loc::File(file_no, start, _) => Loc::File(*file_no, *start, *start),
51 loc => *loc,
52 }
53 }
54
55 #[inline]
57 pub fn end_range(&self) -> Self {
58 match self {
59 Loc::File(file_no, _, end) => Loc::File(*file_no, *end, *end),
60 loc => *loc,
61 }
62 }
63
64 #[track_caller]
70 #[inline]
71 pub fn file_no(&self) -> usize {
72 match self {
73 Loc::File(file_no, _, _) => *file_no,
74 _ => not_a_file(),
75 }
76 }
77
78 #[inline]
80 pub fn try_file_no(&self) -> Option<usize> {
81 match self {
82 Loc::File(file_no, _, _) => Some(*file_no),
83 _ => None,
84 }
85 }
86
87 #[track_caller]
93 #[inline]
94 pub fn start(&self) -> usize {
95 match self {
96 Loc::File(_, start, _) => *start,
97 _ => not_a_file(),
98 }
99 }
100
101 #[track_caller]
107 #[inline]
108 pub fn end(&self) -> usize {
109 match self {
110 Loc::File(_, _, end) => *end,
111 _ => not_a_file(),
112 }
113 }
114
115 #[track_caller]
123 #[inline]
124 pub fn exclusive_end(&self) -> usize {
125 self.end() + 1
126 }
127
128 #[track_caller]
134 #[inline]
135 pub fn use_start_from(&mut self, other: &Loc) {
136 match (self, other) {
137 (Loc::File(_, start, _), Loc::File(_, other_start, _)) => {
138 *start = *other_start;
139 }
140 _ => not_a_file(),
141 }
142 }
143
144 #[track_caller]
150 #[inline]
151 pub fn use_end_from(&mut self, other: &Loc) {
152 match (self, other) {
153 (Loc::File(_, _, end), Loc::File(_, _, other_end)) => {
154 *end = *other_end;
155 }
156 _ => not_a_file(),
157 }
158 }
159
160 #[track_caller]
166 #[inline]
167 pub fn with_start_from(mut self, other: &Self) -> Self {
168 self.use_start_from(other);
169 self
170 }
171
172 #[track_caller]
178 #[inline]
179 pub fn with_end_from(mut self, other: &Self) -> Self {
180 self.use_end_from(other);
181 self
182 }
183
184 #[track_caller]
190 #[inline]
191 pub fn with_start(self, start: usize) -> Self {
192 match self {
193 Self::File(no, _, end) => Self::File(no, start, end),
194 _ => not_a_file(),
195 }
196 }
197
198 #[track_caller]
204 #[inline]
205 pub fn with_end(self, end: usize) -> Self {
206 match self {
207 Self::File(no, start, _) => Self::File(no, start, end),
208 _ => not_a_file(),
209 }
210 }
211
212 #[track_caller]
218 #[inline]
219 pub fn range(self) -> std::ops::Range<usize> {
220 match self {
221 Self::File(_, start, end) => start..end,
222 _ => not_a_file(),
223 }
224 }
225
226 pub fn union(&mut self, other: &Self) {
228 match (self, other) {
229 (Self::File(r_file, r_start, r_end), Self::File(l_file, l_start, l_end)) => {
230 assert_eq!(r_file, l_file, "cannot perform union in different files");
231 *r_start = std::cmp::min(*r_start, *l_start);
232 *r_end = std::cmp::max(*r_end, *l_end);
233 }
234
235 _ => unimplemented!("cannot perform union in non File Loc"),
236 }
237 }
238}
239
240#[derive(Debug, PartialEq, Eq, Clone)]
244#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
245pub struct Identifier {
246 pub loc: Loc,
248 pub name: String,
250}
251
252impl Identifier {
253 pub fn new(s: impl Into<String>) -> Self {
255 Self {
256 loc: Loc::default(),
257 name: s.into(),
258 }
259 }
260}
261
262#[derive(Debug, PartialEq, Eq, Clone)]
266#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
267pub struct IdentifierPath {
268 pub loc: Loc,
270 pub identifiers: Vec<Identifier>,
272}
273
274#[derive(Debug, PartialEq, Eq, Clone)]
278#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
279pub enum Comment {
280 Line(Loc, String),
284
285 Block(Loc, String),
289
290 DocLine(Loc, String),
294
295 DocBlock(Loc, String),
303}
304
305impl Comment {
306 #[inline]
308 pub const fn value(&self) -> &String {
309 match self {
310 Self::Line(_, s) | Self::Block(_, s) | Self::DocLine(_, s) | Self::DocBlock(_, s) => s,
311 }
312 }
313
314 #[inline]
316 pub const fn is_doc(&self) -> bool {
317 matches!(self, Self::DocLine(..) | Self::DocBlock(..))
318 }
319
320 #[inline]
322 pub const fn is_line(&self) -> bool {
323 matches!(self, Self::Line(..) | Self::DocLine(..))
324 }
325
326 #[inline]
328 pub const fn is_block(&self) -> bool {
329 !self.is_line()
330 }
331}
332
333#[derive(Debug, PartialEq, Eq, Clone)]
337#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
338pub struct SourceUnit(pub Vec<SourceUnitPart>);
339
340#[derive(Debug, PartialEq, Eq, Clone)]
342#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
343pub enum SourceUnitPart {
344 PragmaDirective(Box<PragmaDirective>),
350
351 ImportDirective(Import),
353
354 ContractDefinition(Box<ContractDefinition>),
356
357 EnumDefinition(Box<EnumDefinition>),
359
360 StructDefinition(Box<StructDefinition>),
362
363 EventDefinition(Box<EventDefinition>),
365
366 ErrorDefinition(Box<ErrorDefinition>),
368
369 FunctionDefinition(Box<FunctionDefinition>),
371
372 VariableDefinition(Box<VariableDefinition>),
374
375 TypeDefinition(Box<TypeDefinition>),
377
378 Annotation(Box<Annotation>),
380
381 Using(Box<Using>),
383
384 StraySemicolon(Loc),
386}
387
388#[derive(Debug, PartialEq, Eq, Clone)]
390#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
391pub enum Import {
392 Plain(ImportPath, Loc),
394
395 GlobalSymbol(ImportPath, Identifier, Loc),
401
402 Rename(ImportPath, Vec<(Identifier, Option<Identifier>)>, Loc),
404}
405
406#[derive(Debug, PartialEq, Eq, Clone)]
408#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
409pub enum ImportPath {
410 Filename(StringLiteral),
412 Path(IdentifierPath),
414}
415
416impl Import {
417 #[inline]
419 pub const fn literal(&self) -> Option<&StringLiteral> {
420 match self {
421 Self::Plain(ImportPath::Filename(literal), _)
422 | Self::GlobalSymbol(ImportPath::Filename(literal), _, _)
423 | Self::Rename(ImportPath::Filename(literal), _, _) => Some(literal),
424 _ => None,
425 }
426 }
427}
428
429pub type ParameterList = Vec<(Loc, Option<Parameter>)>;
431
432#[derive(Debug, PartialEq, Eq, Clone)]
434#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
435pub enum Type {
436 Address,
438
439 AddressPayable,
441
442 Payable,
446
447 Bool,
449
450 String,
452
453 Int(u16),
455
456 Uint(u16),
458
459 Bytes(u8),
461
462 Rational,
464
465 DynamicBytes,
467
468 Mapping {
470 loc: Loc,
472 key: Box<Expression>,
476 key_name: Option<Identifier>,
478 value: Box<Expression>,
480 value_name: Option<Identifier>,
482 },
483
484 Function {
486 params: ParameterList,
488 attributes: Vec<FunctionAttribute>,
490 returns: Option<(ParameterList, Vec<FunctionAttribute>)>,
492 },
493}
494
495#[derive(Debug, PartialEq, Eq, Clone)]
497#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
498pub enum StorageLocation {
499 Memory(Loc),
501
502 Storage(Loc),
504
505 Calldata(Loc),
507}
508
509#[derive(Debug, PartialEq, Eq, Clone)]
513#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
514pub struct VariableDeclaration {
515 pub loc: Loc,
517 pub ty: Expression,
519 pub storage: Option<StorageLocation>,
521 pub name: Option<Identifier>,
525}
526
527#[derive(Debug, PartialEq, Eq, Clone)]
531#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
532pub struct StructDefinition {
533 pub loc: Loc,
535 pub name: Option<Identifier>,
539 pub fields: Vec<VariableDeclaration>,
541}
542
543#[derive(Debug, PartialEq, Eq, Clone)]
545#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
546pub enum ContractPart {
547 StructDefinition(Box<StructDefinition>),
549
550 EventDefinition(Box<EventDefinition>),
552
553 EnumDefinition(Box<EnumDefinition>),
555
556 ErrorDefinition(Box<ErrorDefinition>),
558
559 VariableDefinition(Box<VariableDefinition>),
561
562 FunctionDefinition(Box<FunctionDefinition>),
564
565 TypeDefinition(Box<TypeDefinition>),
567
568 Annotation(Box<Annotation>),
570
571 Using(Box<Using>),
573
574 StraySemicolon(Loc),
576}
577
578#[derive(Debug, PartialEq, Eq, Clone)]
580#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
581pub enum PragmaDirective {
582 Identifier(Loc, Option<Identifier>, Option<Identifier>),
584 StringLiteral(Loc, Identifier, StringLiteral),
586 Version(Loc, Identifier, Vec<VersionComparator>),
588}
589
590#[derive(Debug, PartialEq, Eq, Clone)]
592#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
593pub enum VersionComparator {
594 Plain {
596 loc: Loc,
598 version: Vec<String>,
600 },
601 Operator {
603 loc: Loc,
605 op: VersionOp,
607 version: Vec<String>,
609 },
610 Or {
612 loc: Loc,
614 left: Box<VersionComparator>,
616 right: Box<VersionComparator>,
618 },
619 Range {
621 loc: Loc,
623 from: Vec<String>,
625 to: Vec<String>,
627 },
628}
629
630#[derive(Debug, PartialEq, Eq, Copy, Clone)]
632#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
633pub enum VersionOp {
634 Exact,
636 Greater,
638 GreaterEq,
640 Less,
642 LessEq,
644 Tilde,
646 Caret,
648 Wildcard,
650}
651
652#[derive(Debug, PartialEq, Eq, Clone)]
654#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
655pub enum UsingList {
656 Library(IdentifierPath),
658
659 Functions(Vec<UsingFunction>),
663
664 Error,
666}
667
668#[derive(Debug, PartialEq, Eq, Clone)]
672#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
673pub struct UsingFunction {
674 pub loc: Loc,
676 pub path: IdentifierPath,
678 pub oper: Option<UserDefinedOperator>,
680}
681
682#[derive(Clone, Copy, Debug, PartialEq, Eq)]
688#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
689pub enum UserDefinedOperator {
690 BitwiseAnd,
692 BitwiseNot,
695 Negate,
699 BitwiseOr,
701 BitwiseXor,
703 Add,
705 Divide,
707 Modulo,
709 Multiply,
711 Subtract,
713 Equal,
715 More,
717 MoreEqual,
719 Less,
721 LessEqual,
723 NotEqual,
725}
726
727impl UserDefinedOperator {
728 #[inline]
730 pub const fn args(&self) -> usize {
731 match self {
732 UserDefinedOperator::BitwiseNot | UserDefinedOperator::Negate => 1,
733 _ => 2,
734 }
735 }
736
737 #[inline]
739 pub const fn is_unary(&self) -> bool {
740 matches!(self, Self::BitwiseNot | Self::Negate)
741 }
742
743 #[inline]
745 pub const fn is_binary(&self) -> bool {
746 !self.is_unary()
747 }
748
749 #[inline]
751 pub const fn is_bitwise(&self) -> bool {
752 matches!(
753 self,
754 Self::BitwiseAnd | Self::BitwiseOr | Self::BitwiseXor | Self::BitwiseNot
755 )
756 }
757
758 #[inline]
760 pub const fn is_arithmetic(&self) -> bool {
761 matches!(
762 self,
763 Self::Add | Self::Subtract | Self::Multiply | Self::Divide | Self::Modulo
764 )
765 }
766
767 #[inline]
769 pub const fn is_comparison(&self) -> bool {
770 matches!(
771 self,
772 Self::Equal
773 | Self::NotEqual
774 | Self::Less
775 | Self::LessEqual
776 | Self::More
777 | Self::MoreEqual
778 )
779 }
780}
781
782#[derive(Debug, PartialEq, Eq, Clone)]
788#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
789pub struct Using {
790 pub loc: Loc,
792 pub list: UsingList,
794 pub ty: Option<Expression>,
798 pub global: Option<Identifier>,
800}
801
802#[derive(Debug, PartialEq, Eq, Clone)]
804#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
805pub enum ContractTy {
806 Abstract(Loc),
808
809 Contract(Loc),
811
812 Interface(Loc),
814
815 Library(Loc),
817}
818
819#[derive(Debug, PartialEq, Eq, Clone)]
826#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
827pub struct Base {
828 pub loc: Loc,
830 pub name: IdentifierPath,
832 pub args: Option<Vec<Expression>>,
834}
835
836#[derive(Debug, PartialEq, Eq, Clone)]
840#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
841pub struct ContractDefinition {
842 pub loc: Loc,
844 pub ty: ContractTy,
846 pub name: Option<Identifier>,
850 pub base: Vec<Base>,
852 pub parts: Vec<ContractPart>,
854}
855
856#[derive(Debug, PartialEq, Eq, Clone)]
860#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
861pub struct EventParameter {
862 pub loc: Loc,
864 pub ty: Expression,
866 pub indexed: bool,
868 pub name: Option<Identifier>,
870}
871
872#[derive(Debug, PartialEq, Eq, Clone)]
876#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
877pub struct EventDefinition {
878 pub loc: Loc,
880 pub name: Option<Identifier>,
884 pub fields: Vec<EventParameter>,
886 pub anonymous: bool,
888}
889
890#[derive(Debug, PartialEq, Eq, Clone)]
894#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
895pub struct ErrorParameter {
896 pub loc: Loc,
898 pub ty: Expression,
900 pub name: Option<Identifier>,
902}
903
904#[derive(Debug, PartialEq, Eq, Clone)]
908#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
909pub struct ErrorDefinition {
910 pub loc: Loc,
912 pub keyword: Expression,
914 pub name: Option<Identifier>,
918 pub fields: Vec<ErrorParameter>,
920}
921
922#[derive(Debug, PartialEq, Eq, Clone)]
926#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
927pub struct EnumDefinition {
928 pub loc: Loc,
930 pub name: Option<Identifier>,
934 pub values: Vec<Option<Identifier>>,
938}
939
940#[derive(Debug, PartialEq, Eq, Clone)]
942#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
943#[repr(u8)] pub enum VariableAttribute {
945 Visibility(Visibility),
949
950 Constant(Loc),
952
953 Immutable(Loc),
955
956 Override(Loc, Vec<IdentifierPath>),
958}
959
960#[derive(Debug, PartialEq, Eq, Clone)]
964#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
965pub struct VariableDefinition {
966 pub loc: Loc,
968 pub ty: Expression,
970 pub attrs: Vec<VariableAttribute>,
972 pub name: Option<Identifier>,
976 pub initializer: Option<Expression>,
978}
979
980#[derive(Debug, PartialEq, Eq, Clone)]
984#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
985pub struct TypeDefinition {
986 pub loc: Loc,
988 pub name: Identifier,
990 pub ty: Expression,
992}
993
994#[derive(Debug, PartialEq, Eq, Clone)]
998#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
999pub struct Annotation {
1000 pub loc: Loc,
1002 pub id: Identifier,
1004 pub value: Option<Expression>,
1006}
1007
1008#[derive(Debug, PartialEq, Eq, Clone)]
1012#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1013pub struct StringLiteral {
1014 pub loc: Loc,
1016 pub unicode: bool,
1018 pub string: String,
1022}
1023
1024#[derive(Debug, PartialEq, Eq, Clone)]
1028#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1029pub struct HexLiteral {
1030 pub loc: Loc,
1032 pub hex: String,
1036}
1037
1038#[derive(Debug, PartialEq, Eq, Clone)]
1042#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1043pub struct NamedArgument {
1044 pub loc: Loc,
1046 pub name: Identifier,
1048 pub expr: Expression,
1050}
1051
1052#[derive(Debug, PartialEq, Eq, Clone)]
1054#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1055pub enum Expression {
1056 PostIncrement(Loc, Box<Expression>),
1058 PostDecrement(Loc, Box<Expression>),
1060 New(Loc, Box<Expression>),
1062 ArraySubscript(Loc, Box<Expression>, Option<Box<Expression>>),
1064 ArraySlice(
1066 Loc,
1067 Box<Expression>,
1068 Option<Box<Expression>>,
1069 Option<Box<Expression>>,
1070 ),
1071 Parenthesis(Loc, Box<Expression>),
1073 MemberAccess(Loc, Box<Expression>, Identifier),
1075 FunctionCall(Loc, Box<Expression>, Vec<Expression>),
1077 FunctionCallBlock(Loc, Box<Expression>, Box<Statement>),
1079 NamedFunctionCall(Loc, Box<Expression>, Vec<NamedArgument>),
1081 Not(Loc, Box<Expression>),
1083 BitwiseNot(Loc, Box<Expression>),
1085 Delete(Loc, Box<Expression>),
1087 PreIncrement(Loc, Box<Expression>),
1089 PreDecrement(Loc, Box<Expression>),
1091 UnaryPlus(Loc, Box<Expression>),
1095 Negate(Loc, Box<Expression>),
1097
1098 Power(Loc, Box<Expression>, Box<Expression>),
1100 Multiply(Loc, Box<Expression>, Box<Expression>),
1102 Divide(Loc, Box<Expression>, Box<Expression>),
1104 Modulo(Loc, Box<Expression>, Box<Expression>),
1106 Add(Loc, Box<Expression>, Box<Expression>),
1108 Subtract(Loc, Box<Expression>, Box<Expression>),
1110 ShiftLeft(Loc, Box<Expression>, Box<Expression>),
1112 ShiftRight(Loc, Box<Expression>, Box<Expression>),
1114 BitwiseAnd(Loc, Box<Expression>, Box<Expression>),
1116 BitwiseXor(Loc, Box<Expression>, Box<Expression>),
1118 BitwiseOr(Loc, Box<Expression>, Box<Expression>),
1120 Less(Loc, Box<Expression>, Box<Expression>),
1122 More(Loc, Box<Expression>, Box<Expression>),
1124 LessEqual(Loc, Box<Expression>, Box<Expression>),
1126 MoreEqual(Loc, Box<Expression>, Box<Expression>),
1128 Equal(Loc, Box<Expression>, Box<Expression>),
1130 NotEqual(Loc, Box<Expression>, Box<Expression>),
1132 And(Loc, Box<Expression>, Box<Expression>),
1134 Or(Loc, Box<Expression>, Box<Expression>),
1136 ConditionalOperator(Loc, Box<Expression>, Box<Expression>, Box<Expression>),
1140 Assign(Loc, Box<Expression>, Box<Expression>),
1142 AssignOr(Loc, Box<Expression>, Box<Expression>),
1144 AssignAnd(Loc, Box<Expression>, Box<Expression>),
1146 AssignXor(Loc, Box<Expression>, Box<Expression>),
1148 AssignShiftLeft(Loc, Box<Expression>, Box<Expression>),
1150 AssignShiftRight(Loc, Box<Expression>, Box<Expression>),
1152 AssignAdd(Loc, Box<Expression>, Box<Expression>),
1154 AssignSubtract(Loc, Box<Expression>, Box<Expression>),
1156 AssignMultiply(Loc, Box<Expression>, Box<Expression>),
1158 AssignDivide(Loc, Box<Expression>, Box<Expression>),
1160 AssignModulo(Loc, Box<Expression>, Box<Expression>),
1162
1163 BoolLiteral(Loc, bool),
1165 NumberLiteral(Loc, String, String, Option<Identifier>),
1167 RationalNumberLiteral(Loc, String, String, String, Option<Identifier>),
1169 HexNumberLiteral(Loc, String, Option<Identifier>),
1171 StringLiteral(Vec<StringLiteral>),
1173 Type(Loc, Type),
1175 HexLiteral(Vec<HexLiteral>),
1177 AddressLiteral(Loc, String),
1183 Variable(Identifier),
1185 List(Loc, ParameterList),
1187 ArrayLiteral(Loc, Vec<Expression>),
1189}
1190
1191macro_rules! expr_components {
1193 ($s:ident) => {
1194 match $s {
1195 PostDecrement(_, expr) | PostIncrement(_, expr) => (Some(expr), None),
1197
1198 Not(_, expr)
1200 | BitwiseNot(_, expr)
1201 | New(_, expr)
1202 | Delete(_, expr)
1203 | UnaryPlus(_, expr)
1204 | Negate(_, expr)
1205 | PreDecrement(_, expr)
1206 | Parenthesis(_, expr)
1207 | PreIncrement(_, expr) => (None, Some(expr)),
1208
1209 Power(_, left, right)
1211 | Multiply(_, left, right)
1212 | Divide(_, left, right)
1213 | Modulo(_, left, right)
1214 | Add(_, left, right)
1215 | Subtract(_, left, right)
1216 | ShiftLeft(_, left, right)
1217 | ShiftRight(_, left, right)
1218 | BitwiseAnd(_, left, right)
1219 | BitwiseXor(_, left, right)
1220 | BitwiseOr(_, left, right)
1221 | Less(_, left, right)
1222 | More(_, left, right)
1223 | LessEqual(_, left, right)
1224 | MoreEqual(_, left, right)
1225 | Equal(_, left, right)
1226 | NotEqual(_, left, right)
1227 | And(_, left, right)
1228 | Or(_, left, right)
1229 | Assign(_, left, right)
1230 | AssignOr(_, left, right)
1231 | AssignAnd(_, left, right)
1232 | AssignXor(_, left, right)
1233 | AssignShiftLeft(_, left, right)
1234 | AssignShiftRight(_, left, right)
1235 | AssignAdd(_, left, right)
1236 | AssignSubtract(_, left, right)
1237 | AssignMultiply(_, left, right)
1238 | AssignDivide(_, left, right)
1239 | AssignModulo(_, left, right) => (Some(left), Some(right)),
1240
1241 MemberAccess(..)
1243 | ConditionalOperator(..)
1244 | ArraySubscript(..)
1245 | ArraySlice(..)
1246 | FunctionCall(..)
1247 | FunctionCallBlock(..)
1248 | NamedFunctionCall(..)
1249 | BoolLiteral(..)
1250 | NumberLiteral(..)
1251 | RationalNumberLiteral(..)
1252 | HexNumberLiteral(..)
1253 | StringLiteral(..)
1254 | Type(..)
1255 | HexLiteral(..)
1256 | AddressLiteral(..)
1257 | Variable(..)
1258 | List(..)
1259 | ArrayLiteral(..) => (None, None),
1260 }
1261 };
1262}
1263
1264impl Expression {
1265 #[inline]
1267 pub fn remove_parenthesis(&self) -> &Expression {
1268 if let Expression::Parenthesis(_, expr) = self {
1269 expr
1270 } else {
1271 self
1272 }
1273 }
1274
1275 pub fn strip_parentheses(&self) -> &Expression {
1277 match self {
1278 Expression::Parenthesis(_, expr) => expr.strip_parentheses(),
1279 _ => self,
1280 }
1281 }
1282
1283 #[inline]
1309 pub fn components(&self) -> (Option<&Self>, Option<&Self>) {
1310 use Expression::*;
1311 expr_components!(self)
1312 }
1313
1314 #[inline]
1318 pub fn components_mut(&mut self) -> (Option<&mut Self>, Option<&mut Self>) {
1319 use Expression::*;
1320 expr_components!(self)
1321 }
1322
1323 #[inline]
1325 pub const fn is_unsplittable(&self) -> bool {
1326 use Expression::*;
1327 matches!(
1328 self,
1329 BoolLiteral(..)
1330 | NumberLiteral(..)
1331 | RationalNumberLiteral(..)
1332 | HexNumberLiteral(..)
1333 | StringLiteral(..)
1334 | HexLiteral(..)
1335 | AddressLiteral(..)
1336 | Variable(..)
1337 )
1338 }
1339
1340 #[inline]
1342 pub const fn has_space_around(&self) -> bool {
1343 use Expression::*;
1344 !matches!(
1345 self,
1346 PostIncrement(..)
1347 | PreIncrement(..)
1348 | PostDecrement(..)
1349 | PreDecrement(..)
1350 | Not(..)
1351 | BitwiseNot(..)
1352 | UnaryPlus(..)
1353 | Negate(..)
1354 )
1355 }
1356
1357 pub fn is_literal(&self) -> bool {
1359 matches!(
1360 self,
1361 Expression::AddressLiteral(..)
1362 | Expression::HexLiteral(..)
1363 | Expression::BoolLiteral(..)
1364 | Expression::NumberLiteral(..)
1365 | Expression::ArrayLiteral(..)
1366 | Expression::HexNumberLiteral(..)
1367 | Expression::RationalNumberLiteral(..)
1368 | Expression::StringLiteral(..)
1369 )
1370 }
1371}
1372
1373#[derive(Debug, PartialEq, Eq, Clone)]
1377#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1378pub struct Parameter {
1379 pub loc: Loc,
1381 pub annotation: Option<Annotation>,
1383 pub ty: Expression,
1385 pub storage: Option<StorageLocation>,
1387 pub name: Option<Identifier>,
1389}
1390
1391#[derive(Debug, PartialEq, Eq, Clone)]
1393#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1394pub enum Mutability {
1395 Pure(Loc),
1397
1398 View(Loc),
1400
1401 Constant(Loc),
1403
1404 Payable(Loc),
1406}
1407
1408#[derive(Debug, PartialEq, Eq, Clone)]
1412#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1413#[repr(u8)] pub enum Visibility {
1415 External(Option<Loc>),
1417
1418 Public(Option<Loc>),
1420
1421 Internal(Option<Loc>),
1423
1424 Private(Option<Loc>),
1426}
1427
1428#[derive(Debug, PartialEq, Eq, Clone)]
1430#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1431#[repr(u8)] pub enum FunctionAttribute {
1433 Visibility(Visibility),
1435
1436 Mutability(Mutability),
1438
1439 Virtual(Loc),
1441
1442 Immutable(Loc),
1444
1445 Override(Loc, Vec<IdentifierPath>),
1447
1448 BaseOrModifier(Loc, Base),
1450
1451 Error(Loc),
1453}
1454
1455#[derive(Debug, PartialEq, Eq, Clone, Copy)]
1457#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1458pub enum FunctionTy {
1459 Constructor,
1461
1462 Function,
1464
1465 Fallback,
1467
1468 Receive,
1470
1471 Modifier,
1473}
1474
1475#[derive(Debug, PartialEq, Eq, Clone)]
1479#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1480pub struct FunctionDefinition {
1481 pub loc_prototype: Loc,
1483 pub loc: Loc,
1485 pub ty: FunctionTy,
1487 pub name: Option<Identifier>,
1491 pub name_loc: Loc,
1493 pub params: ParameterList,
1495 pub attributes: Vec<FunctionAttribute>,
1497 pub return_not_returns: Option<Loc>,
1499 pub returns: ParameterList,
1501 pub body: Option<Statement>,
1505}
1506
1507impl FunctionDefinition {
1508 #[inline]
1510 pub fn is_void(&self) -> bool {
1511 self.returns.is_empty()
1512 }
1513
1514 #[inline]
1516 pub fn is_empty(&self) -> bool {
1517 self.body.as_ref().map_or(true, Statement::is_empty)
1518 }
1519
1520 #[inline]
1522 pub fn sort_attributes(&mut self) {
1523 self.attributes.sort();
1526 }
1527}
1528
1529#[derive(Debug, PartialEq, Eq, Clone)]
1531#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1532#[allow(clippy::large_enum_variant, clippy::type_complexity)]
1533pub enum Statement {
1534 Block {
1536 loc: Loc,
1538 unchecked: bool,
1540 statements: Vec<Statement>,
1542 },
1543 Assembly {
1545 loc: Loc,
1547 dialect: Option<StringLiteral>,
1549 flags: Option<Vec<StringLiteral>>,
1551 block: YulBlock,
1553 },
1554 Args(Loc, Vec<NamedArgument>),
1556 If(Loc, Expression, Box<Statement>, Option<Box<Statement>>),
1560 While(Loc, Expression, Box<Statement>),
1564 Expression(Loc, Expression),
1566 VariableDefinition(Loc, VariableDeclaration, Option<Expression>),
1568 For(
1572 Loc,
1573 Option<Box<Statement>>,
1574 Option<Box<Expression>>,
1575 Option<Box<Expression>>,
1576 Option<Box<Statement>>,
1577 ),
1578 DoWhile(Loc, Box<Statement>, Expression),
1582 Continue(Loc),
1584 Break(Loc),
1586 Return(Loc, Option<Expression>),
1588 Revert(Loc, Option<IdentifierPath>, Vec<Expression>),
1590 RevertNamedArgs(Loc, Option<IdentifierPath>, Vec<NamedArgument>),
1592 Emit(Loc, Expression),
1596 Try(
1600 Loc,
1601 Expression,
1602 Option<(ParameterList, Box<Statement>)>,
1603 Vec<CatchClause>,
1604 ),
1605 Error(Loc),
1607}
1608
1609impl Statement {
1610 #[inline]
1612 pub fn is_empty(&self) -> bool {
1613 match self {
1614 Self::Block { statements, .. } => statements.is_empty(),
1615 Self::Assembly { block, .. } => block.is_empty(),
1616 Self::Args(_, args) => args.is_empty(),
1617 _ => false,
1618 }
1619 }
1620}
1621
1622#[derive(Debug, PartialEq, Eq, Clone)]
1624#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1625pub enum CatchClause {
1626 Simple(Loc, Option<Parameter>, Statement),
1628
1629 Named(Loc, Identifier, Parameter, Statement),
1631}
1632
1633#[derive(Debug, PartialEq, Eq, Clone)]
1635#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1636pub enum YulStatement {
1637 Assign(Loc, Vec<YulExpression>, YulExpression),
1639 VariableDeclaration(Loc, Vec<YulTypedIdentifier>, Option<YulExpression>),
1641 If(Loc, YulExpression, YulBlock),
1643 For(YulFor),
1645 Switch(YulSwitch),
1647 Leave(Loc),
1649 Break(Loc),
1651 Continue(Loc),
1653 Block(YulBlock),
1655 FunctionDefinition(Box<YulFunctionDefinition>),
1657 FunctionCall(Box<YulFunctionCall>),
1659 Error(Loc),
1661}
1662
1663#[derive(Debug, PartialEq, Eq, Clone)]
1673#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1674pub struct YulSwitch {
1675 pub loc: Loc,
1677 pub condition: YulExpression,
1679 pub cases: Vec<YulSwitchOptions>,
1681 pub default: Option<YulSwitchOptions>,
1683}
1684
1685#[derive(Debug, PartialEq, Eq, Clone)]
1689#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1690pub struct YulFor {
1691 pub loc: Loc,
1693 pub init_block: YulBlock,
1695 pub condition: YulExpression,
1697 pub post_block: YulBlock,
1699 pub execution_block: YulBlock,
1701}
1702
1703#[derive(Debug, PartialEq, Eq, Clone)]
1707#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1708pub struct YulBlock {
1709 pub loc: Loc,
1711 pub statements: Vec<YulStatement>,
1713}
1714
1715impl YulBlock {
1716 #[inline]
1718 pub fn is_empty(&self) -> bool {
1719 self.statements.is_empty()
1720 }
1721}
1722
1723#[derive(Debug, PartialEq, Eq, Clone)]
1725#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1726pub enum YulExpression {
1727 BoolLiteral(Loc, bool, Option<Identifier>),
1729 NumberLiteral(Loc, String, String, Option<Identifier>),
1731 HexNumberLiteral(Loc, String, Option<Identifier>),
1733 HexStringLiteral(HexLiteral, Option<Identifier>),
1735 StringLiteral(StringLiteral, Option<Identifier>),
1737 Variable(Identifier),
1739 FunctionCall(Box<YulFunctionCall>),
1741 SuffixAccess(Loc, Box<YulExpression>, Identifier),
1743}
1744
1745#[derive(Debug, PartialEq, Eq, Clone)]
1749#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1750pub struct YulTypedIdentifier {
1751 pub loc: Loc,
1753 pub id: Identifier,
1755 pub ty: Option<Identifier>,
1757}
1758
1759#[derive(Debug, PartialEq, Eq, Clone)]
1763#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1764pub struct YulFunctionDefinition {
1765 pub loc: Loc,
1767 pub id: Identifier,
1769 pub params: Vec<YulTypedIdentifier>,
1771 pub returns: Vec<YulTypedIdentifier>,
1773 pub body: YulBlock,
1775}
1776
1777impl YulFunctionDefinition {
1778 #[inline]
1780 pub fn is_void(&self) -> bool {
1781 self.returns.is_empty()
1782 }
1783
1784 #[inline]
1786 pub fn is_empty(&self) -> bool {
1787 self.body.is_empty()
1788 }
1789}
1790
1791#[derive(Debug, PartialEq, Eq, Clone)]
1795#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1796pub struct YulFunctionCall {
1797 pub loc: Loc,
1799 pub id: Identifier,
1801 pub arguments: Vec<YulExpression>,
1803}
1804
1805#[derive(Debug, PartialEq, Eq, Clone)]
1807#[cfg_attr(feature = "pt-serde", derive(Serialize, Deserialize))]
1808pub enum YulSwitchOptions {
1809 Case(Loc, YulExpression, YulBlock),
1811 Default(Loc, YulBlock),
1813}