-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathYoloGit.jl
252 lines (229 loc) · 10.5 KB
/
YoloGit.jl
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
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
#Author: Yavuz Faruk Bakman
#Date: 15/08/2019
#Description: Tiny Yolo V2 implementation by Knet framework.
#Currently, it uses pre-trained weigths and doesn't train the model
#Load necessary packages
using Pkg; for p in ("Knet","Random","Glob","FileIO","DelimitedFiles","OffsetArrays","Images",
"ImageDraw","ImageMagick","ImageFiltering","ImageTransformations","Colors","FreeTypeAbstraction","QuartzImageIO","LightXML","ArgParse");
haskey(Pkg.installed(),p) || Pkg.add(p); end
#Include related files.
include("inputprocess.jl") #containing functions preparing input
include("outprocess.jl") # containing functions preparing output
include("loadweights.jl") # containing functions loading weights to model
using Knet: Knet, progress, progress!, gpu, KnetArray,
relu,minibatch,conv4, pool, softmax,minibatch,progress
using Random, Glob, FileIO, DelimitedFiles, OffsetArrays
using Images, ImageDraw, ImageFiltering, ImageTransformations, Colors
using FreeTypeAbstraction
using LightXML
using ImageMagick,ArgParse
face = newface("DroidSansMono.ttf") #Font type
xtype=(Knet.gpu()>=0 ? Knet.KnetArray{Float32} : Array{Float32})#if gpu exists run on gpu
#Yolo V2 pre-trained boxes width and height
anchors = [(1.08,1.19), (3.42,4.41), (6.63,11.38), (9.42,5.11), (16.62,10.52)]
#20 classes on Voc dataset
numClass = 20
MINIBATCH_SIZE = 1
WEIGHTS_FILE = "yolov2-tiny-voc.weights" #Pre-trained weights data
ACC_INPUT = "VOCdevkit/VOC2007/JPEGImages" #Input directory for accuracy calculation
ACC_OUT = "VOCdevkit/VOC2007/Annotations" #location of objects as Xml file for accuracy calculation
INPUT = "Input" #Input directory to create output and save
EXAMPLE_INPUT = "example.jpg" #One input for display
# 2 dictionaries to access number<->class by O(1)
namesdic = Dict("aeroplane"=>1,"bicycle"=>2,"bird"=>3, "boat"=>4,
"bottle"=>5,"bus"=>6,"car"=>7,"cat"=>8,"chair"=>9,
"cow"=>10,"diningtable"=>11,"dog"=>12,"horse"=>13,"motorbike"=>14,
"person"=>15,"pottedplant"=>16,"sheep"=>17,"sofa"=>18,"train"=>19,"tvmonitor"=>20)
numsdic = Dict(1=>"aeroplane",2=>"bicycle",3=>"bird", 4=>"boat",
5=>"bottle",6=>"bus",7=>"car",8=>"cat",9=>"chair",
10=>"cow",11=>"diningtable",12=>"dog",13=>"horse",14=>"motorbike",
15=>"person",16=>"pottedplant",17=>"sheep",18=>"sofa",19=>"train",20=>"tvmonitor")
#Store total amount of the objects to calculate accuracy
totaldic = Dict("aeroplane"=>0,"bicycle"=>0,"bird"=>0, "boat"=>0,
"bottle"=>0,"bus"=>0,"car"=>0,"cat"=>0,"chair"=>0,
"cow"=>0,"diningtable"=>0,"dog"=>0,"horse"=>0,"motorbike"=>0,
"person"=>0,"pottedplant"=>0,"sheep"=>0,"sofa"=>0,"train"=>0,"tvmonitor"=>0)
#Define sigmoid function
sigmoid(x) = 1.0 / (1.0 .+ exp(-x))
#Define Chain
mutable struct Chain
layers
Chain(layers...) = new(layers)
end
mutable struct Conv; w; b; stride; padding; f; end #Define convolutional layer
mutable struct YoloPad; w; end #Define Yolo padding layer (assymetric padding).
struct Pool; size; stride; pad; end # Define pool layer
YoloPad(w1::Int,w2::Int,cx::Int,cy::Int) = YoloPad(zeros(Float32,w1,w2,cx,cy))#Constructor for Yolopad
Conv(w1::Int,w2::Int,cx::Int,cy::Int,st,pd,f) = Conv(randn(Float32,w1,w2,cx,cy),randn(Float32,1,1,cy,1),st,pd,f)#Constructor for convolutional layer
#Assymetric padding function
function(y::YoloPad)(x)
x = reshape(x,14,14,1,512*MINIBATCH_SIZE)
return reshape(conv4(y.w,x; stride = 1),13,13,512,MINIBATCH_SIZE)
end
(p::Pool)(x) = pool(x; window = p.size, stride = p.stride, padding=p.pad) #pool function
(c::Chain)(x) = (for l in c.layers; x = l(x); end; x) #chain function
(c::Conv)(x) = c.f.(conv4(c.w,x; stride = c.stride, padding = c.padding) .+ c.b) #convolutional layer function
#leaky function
function leaky(x)
return max(Float32(0.1)*x,x)
end
#Tiny Yolo V2 model configuration
model = Chain(Conv(3,3,3,16,1,1,leaky),
Pool(2,2,0),
Conv(3,3,16,32,1,1,leaky),
Pool(2,2,0),
Conv(3,3,32,64,1,1,leaky),
Pool(2,2,0),
Conv(3,3,64,128,1,1,leaky),
Pool(2,2,0),
Conv(3,3,128,256,1,1,leaky),
Pool(2,2,0),
Conv(3,3,256,512,1,1,leaky),
Pool(2,1,1),
YoloPad(2,2,1,1),
Conv(3,3,512,1024,1,1,leaky),
Conv(3,3,1024,1024,1,1,leaky),
Conv(1,1,1024,125,1,0,identity))
#post processing function.
#Confidence score threshold to select correct predictions. Recommended : 0.3
#IoU threshold to remove unnecessary predictions: Recommended:0.3
function postprocessing(out,confth,iouth)
out = Array{Float32,4}(out)
result = []
RATE = 32
for cy in 1:13
for cx in 1:13
for b in 1:5
channel = (b-1)*(numClass + 5)
tx = out[cy,cx,channel+1,1]
ty = out[cy,cx,channel+2,1]
tw = out[cy,cx,channel+3,1]
th = out[cy,cx,channel+4,1]
tc = out[cy,cx,channel+5,1]
x = (sigmoid(tx) + cx-1) * RATE
y = (sigmoid(ty) + cy-1) * RATE
w = exp(tw) * anchors[b][1] * RATE
h = exp(th) * anchors[b][2] * RATE
conf = sigmoid(tc)
classScores = out[cy,cx,channel+6:channel+25,1]
classScores = softmax(classScores)
classNo = argmax(classScores)
bestScore = classScores[classNo]
classConfidenceScore = conf*bestScore
if classConfidenceScore > confth
p = (max(0.0,x-w/2),max(0.0,y-h/2),min(w,416.0),min(h,416.0),classNo,classConfidenceScore)
push!(result,p)
end
end
end
end
result = nonmaxsupression(result,iouth)
return result
end
#It removes the predictions overlapping.
function nonmaxsupression(results,iouth)
sort!(results, by = x ->x[6],rev=true)
for i in 1:length(results)
k = i+1
while k <= length(results)
if ioumatch(results[i][1],results[i][2],results[i][3],results[i][4],
results[k][1],results[k][2],results[k][3],results[k][4]) > iouth && results[i][5] == results[k][5]
deleteat!(results,k)
k = k - 1
end
k = k+1
end
end
return results
end
#It calculates IoU score (overlapping rate)
function ioumatch(x1,y1,w1,h1,x2,y2,w2,h2)
r1 = x1 + w1
l1 = x1
t1 = y1
b1 = y1 + h1
r2 = x2 + w2
l2 = x2
t2 = y2
b2 = y2 + h2
a = min(r1,r2)
b = max(t1,t2)
c = max(l1,l2)
d = min(b1,b2)
if d < b
return 0
end
intersec = (d-b)*(a-c)
return intersec/(w1*h1+w2*h2-intersec)
end
function main(args=ARGS)
s = ArgParseSettings()
s.description="YoloGit.jl Yavuz Bakman,2019. Tiny Yolo V2 implementation by Knet framework"
s.exc_handler=ArgParse.debug_handler
@add_arg_table s begin
("--iou"; arg_type=Float32; default=Float32(0.3); help="If two predictions overlap more than this threshold, one of them is removed")
("--iouth"; arg_type=Float32; default=Float32(0.5); help="The threshold for accuracy calculation. If prediction and ground truth overlap more than this threshold, the prediction is counted as true positive, otherwise false positive")
("--confth"; arg_type=Float32; default=Float32(0.3); help="The threshold for confidence score. If one prediction's score is more than this threshold, It is taken")
("--record"; arg_type= Bool; default=false; help="Sets whether output is saved")
("--directory"; arg_type=String; default="Output"; help="The location for saving the output")
end
isa(args, AbstractString) && (args=split(args))
if in("--help", args) || in("-h", args)
ArgParse.show_help(s; exit_when_done=false)
return
end
println(s.description)
#Load pre-trained weights into the model
f = open(WEIGHTS_FILE)
getweights(model,f)
close(f)
o = parse_args(args[2:end], s; as_symbols=true)
if in("accuracy", args)
#=User guide
inputandlabelsdir => takes Voc labels and inputs folder location respectively and returns 2 arrays
images directories and their labels' directories.
prepareinputlabels => takes images and labels directories and returns 3 arrays
input as 416*416*3*totalImage and
labels as tupple of arrays.
tupples are designed as (ImageWidth, ImageHeight,[x,y,objectWidth,objectHeight],[x,y,objectWidth,objectHeight]..)
images are padded version of given images.
inputdir => takes the directory of input for saving output and returns array of directories of the images
prepareinput => takes the array of directories of the images and returns 416*416*3*totalImage. =#
#prepare data for accuracy
images,labels = inputandlabelsdir(ACC_OUT,ACC_INPUT)
inp,out,imgs = prepareinputlabels(images,labels)
print("input for accuracy: ")
println(summary(inp))
#Minibatching process
accdata = minibatch(inp,out,MINIBATCH_SIZE;xtype = xtype)
AP = accuracy(model,accdata,0.0,o[:iou],o[:iouth])
display(AP)
print("Mean average precision: ")
println(calculatemean(AP))
if o[:record] == true
drawdata = minibatch(inp,imgs,MINIBATCH_SIZE; xtype = xtype)
#output of Voc dataset.
#return output as [ [(x,y,width,height,classNumber,confidenceScore),(x,y,width,height,classNumber,confidenceScore)...] ,[(x,y,width,height,classNumber,confidenceScore),(x,y,width,height,classNumber,confidenceScore)..],...]
#save the output images into given location
result = saveoutput(model,drawdata,o[:confth],o[:iou]; record = true, location = "VocResult")
end
end
if in("display", args)
#Display one test image
displaytest(EXAMPLE_INPUT,model; record = o[:record])
end
if in("saveout", args)
#prepare data for saving process
indir = inputdir(INPUT)
inp,images = prepareinput(indir)
print("input for saving: ")
println(summary(inp))
#Minibatching process
savedata = minibatch(inp,images,MINIBATCH_SIZE; xtype = xtype)
#output of given input.
#return output as same with above example
#It also saves the result of the images into output folder.
@time result2 = saveoutput(model,savedata,0.3,0.3; record = o[:record], location = o[:directory])
end
end
PROGRAM_FILE == "YoloGit.jl" && main(ARGS)