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
use pnets::standard::Net;
use pnets::{arc, PlaceId, TransitionId};
use crate::modifications::{Agglomeration, Modification};
use crate::reducers::reduce::{ConservativeReduce, PlaceReduce};
use crate::reducers::Reduce;
pub struct WeightSimplification;
impl ConservativeReduce<Net> for WeightSimplification {}
impl Reduce<Net> for WeightSimplification {
fn reduce(net: &mut Net, modifications: &mut Vec<Modification>) {
for tr in (0..net.places.len()).map(|v| PlaceId::from(v)) {
Self::place_reduce(net, tr, modifications);
}
}
}
impl PlaceReduce<Net> for WeightSimplification {
fn place_reduce(net: &mut Net, pl: PlaceId, modifications: &mut Vec<Modification>) {
if !net[pl].deleted
&& net[pl].initial > 1
&& net[pl]
.produced_by
.iter()
.all(|(_, w)| *w == net[pl].initial)
&& net[pl]
.consumed_by
.iter()
.all(|(_, w)| *w == net[pl].initial)
{
let new_pl = net.create_place();
net[new_pl].initial = 1;
let consume_transitions: Vec<TransitionId> =
net[pl].consumed_by.iter().map(|(tr, _)| *tr).collect();
for tr in consume_transitions {
net.add_arc(arc::Kind::Consume(new_pl, tr, 1)).unwrap();
}
let produce_transitions: Vec<TransitionId> =
net[pl].produced_by.iter().map(|(tr, _)| *tr).collect();
for tr in produce_transitions {
net.add_arc(arc::Kind::Produce(new_pl, tr, 1)).unwrap();
}
net.delete_place(pl);
modifications.push(Modification::Agglomeration(Agglomeration {
deleted_places: vec![(pl, 1)],
new_place: new_pl,
constant: 0,
factor: net[pl].initial as isize,
}));
}
}
}