-
Notifications
You must be signed in to change notification settings - Fork 470
/
Copy pathmodel.rs
114 lines (100 loc) · 3.32 KB
/
model.rs
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
109
110
111
112
113
114
use burn::{
nn::{
conv::{Conv2d, Conv2dConfig},
pool::{MaxPool2d, MaxPool2dConfig},
Dropout, DropoutConfig, Linear, LinearConfig, PaddingConfig2d, Relu,
},
prelude::*,
};
/// Basic convolutional neural network with VGG-style blocks.
//
// VGG block
// ┌────────────────────┐
// │ 3x3 conv │
// │ ↓ │
// │ activation │
// │ ↓ │
// │ 3x3 conv │
// │ ↓ │
// │ activation │
// │ ↓ │
// │ maxpool │
// └────────────────────┘
#[derive(Module, Debug)]
pub struct Cnn<B: Backend> {
activation: Relu,
dropout: Dropout,
pool: MaxPool2d,
conv1: Conv2d<B>,
conv2: Conv2d<B>,
conv3: Conv2d<B>,
conv4: Conv2d<B>,
conv5: Conv2d<B>,
conv6: Conv2d<B>,
fc1: Linear<B>,
fc2: Linear<B>,
}
impl<B: Backend> Cnn<B> {
pub fn new(num_classes: usize, device: &Device<B>) -> Self {
let conv1 = Conv2dConfig::new([3, 32], [3, 3])
.with_padding(PaddingConfig2d::Same)
.init(device);
let conv2 = Conv2dConfig::new([32, 32], [3, 3])
.with_padding(PaddingConfig2d::Same)
.init(device);
let conv3 = Conv2dConfig::new([32, 64], [3, 3])
.with_padding(PaddingConfig2d::Same)
.init(device);
let conv4 = Conv2dConfig::new([64, 64], [3, 3])
.with_padding(PaddingConfig2d::Same)
.init(device);
let conv5 = Conv2dConfig::new([64, 128], [3, 3])
.with_padding(PaddingConfig2d::Same)
.init(device);
let conv6 = Conv2dConfig::new([128, 128], [3, 3])
.with_padding(PaddingConfig2d::Same)
.init(device);
let pool = MaxPool2dConfig::new([2, 2]).with_strides([2, 2]).init();
let fc1 = LinearConfig::new(2048, 128).init(device);
let fc2 = LinearConfig::new(128, num_classes).init(device);
let dropout = DropoutConfig::new(0.3).init();
Self {
activation: Relu::new(),
dropout,
pool,
conv1,
conv2,
conv3,
conv4,
conv5,
conv6,
fc1,
fc2,
}
}
pub fn forward(&self, x: Tensor<B, 4>) -> Tensor<B, 2> {
let x = self.conv1.forward(x);
let x = self.activation.forward(x);
let x = self.conv2.forward(x);
let x = self.activation.forward(x);
let x = self.pool.forward(x);
let x = self.dropout.forward(x);
let x = self.conv3.forward(x);
let x = self.activation.forward(x);
let x = self.conv4.forward(x);
let x = self.activation.forward(x);
let x = self.pool.forward(x);
let x = self.dropout.forward(x);
let x = self.conv5.forward(x);
let x = self.activation.forward(x);
let x = self.conv6.forward(x);
let x = self.activation.forward(x);
let x = self.pool.forward(x);
let x = self.dropout.forward(x);
let x = x.flatten(1, 3);
let x = self.fc1.forward(x);
let x = self.activation.forward(x);
let x = self.dropout.forward(x);
self.fc2.forward(x)
}
}