1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
use super::rules::RuleBase;
#[derive(Copy, Clone, Debug, Hash, PartialEq, Eq)]
pub enum RoutingLayerType {
Cut,
Routing
}
pub struct RoutingLayer<LayerId> {
id: LayerId,
layer_type: RoutingLayerType
}
impl<LayerId> RoutingLayer<LayerId> {
pub fn new(id: LayerId, layer_type: RoutingLayerType) -> Self {
Self {
id,
layer_type
}
}
pub fn as_id(&self) -> &LayerId {
&self.id
}
pub fn id(self) -> LayerId {
self.id
}
pub fn layer_type(&self) -> RoutingLayerType {
self.layer_type
}
pub fn is_via_layer(&self) -> bool {
match self.layer_type() {
RoutingLayerType::Cut => true,
RoutingLayerType::Routing => false,
}
}
pub fn is_metal_layer(&self) -> bool {
match self.layer_type() {
RoutingLayerType::Cut => false,
RoutingLayerType::Routing => true,
}
}
}
pub trait RoutingLayerStack: RuleBase {
fn layer_stack(&self) -> Vec<RoutingLayer<Self::LayerId>>;
fn routing_layer_stack(&self) -> Vec<Self::LayerId> {
self.layer_stack().into_iter()
.filter(|l| l.is_metal_layer())
.map(|l| l.id())
.collect()
}
fn via_layer_stack(&self) -> Vec<Self::LayerId> {
self.layer_stack().into_iter()
.filter(|l| l.is_via_layer())
.map(|l| l.id())
.collect()
}
fn get_upper_metal_layer(&self, layer: &Self::LayerId) -> Option<Self::LayerId> {
self.layer_stack().into_iter()
.skip_while(|l| l.as_id() != layer)
.skip(1)
.find(|l| l.is_metal_layer())
.map(|l| l.id())
}
fn get_lower_metal_layer(&self, layer: &Self::LayerId) -> Option<Self::LayerId> {
self.layer_stack().into_iter().rev()
.skip_while(|l| l.as_id() != layer)
.skip(1)
.find(|l| l.is_metal_layer())
.map(|l| l.id())
}
}