-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathkuwahara-optimized.go
104 lines (95 loc) · 2.99 KB
/
kuwahara-optimized.go
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
package filters
import (
"math"
"runtime"
"sync"
"time"
)
func Kuwahara(path string, radius uint) {
img, format, openMS, convertMS := open(path)
now := math.Round(float64(time.Now().UnixNano()) / 1000000)
width, height := img.Rect.Max.X, img.Rect.Max.Y
radiusInt := int(radius)
result := make([]uint8, len(img.Pix))
pixLen := len(img.Pix)
threads := runtime.NumCPU()
pixPerThread := getPixPerThread(pixLen, threads)
var wg sync.WaitGroup
processing := func(thread int) {
defer wg.Done()
startIndex := pixPerThread * thread
endIndex := clampMax(startIndex+pixPerThread, pixLen)
for i := startIndex; i < endIndex; i += 4 {
apertureMinX := [4]int{-radiusInt, 0, -radiusInt, 0}
apertureMaxX := [4]int{0, radiusInt, 0, radiusInt}
apertureMinY := [4]int{-radiusInt, -radiusInt, 0, 0}
apertureMaxY := [4]int{0, 0, radiusInt, radiusInt}
x, y := getCoordinates(i/4, width)
rValues := [4]int{0, 0, 0, 0}
gValues := [4]int{0, 0, 0, 0}
bValues := [4]int{0, 0, 0, 0}
maxRValue := [4]int{0, 0, 0, 0}
maxGValue := [4]int{0, 0, 0, 0}
maxBValue := [4]int{0, 0, 0, 0}
minRValue := [4]int{255, 255, 255, 255}
minGValue := [4]int{255, 255, 255, 255}
minBValue := [4]int{255, 255, 255, 255}
pixelsCount := [4]int{0, 0, 0, 0}
for i := 0; i < 4; i += 1 {
x2s, x2e := getAperture(x, width, apertureMinX[i], apertureMaxX[i])
y2s, y2e := getAperture(y, height, apertureMinY[i], apertureMaxY[i])
for x2 := x2s; x2 < x2e; x2 += 1 {
for y2 := y2s; y2 < y2e; y2 += 1 {
px := getPixel(x2, y2, width)
r, g, b := img.Pix[px], img.Pix[px+1], img.Pix[px+2]
rValues[i] += int(r)
gValues[i] += int(g)
bValues[i] += int(b)
if int(r) > maxRValue[i] {
maxRValue[i] = int(r)
} else if int(r) < minRValue[i] {
minRValue[i] = int(r)
}
if int(g) > maxGValue[i] {
maxGValue[i] = int(g)
} else if int(g) < minGValue[i] {
minGValue[i] = int(g)
}
if int(b) > maxBValue[i] {
maxBValue[i] = int(b)
} else if int(b) < minBValue[i] {
minBValue[i] = int(b)
}
pixelsCount[i] += 1
}
}
}
j := 0
minDifference := 10000
for i := 0; i < 4; i += 1 {
cdR := maxRValue[i] - minRValue[i]
cdG := maxGValue[i] - minGValue[i]
cdB := maxBValue[i] - minBValue[i]
CurrentDifference := cdR + cdG + cdB
if CurrentDifference < minDifference && pixelsCount[i] > 0 {
j = i
minDifference = CurrentDifference
}
}
result[i] = uint8(rValues[j] / pixelsCount[j])
result[i+1] = uint8(gValues[j] / pixelsCount[j])
result[i+2] = uint8(bValues[j] / pixelsCount[j])
result[i+3] = img.Pix[i+3]
}
}
for t := 0; t < threads; t += 1 {
wg.Add(1)
go processing(t)
}
wg.Wait()
img.Pix = result
processMS := int(math.Round(float64(time.Now().UnixNano())/1000000) - now)
saveMS := save(img, format)
sum := openMS + convertMS + processMS + saveMS
println("open", openMS, "convert", convertMS, "process", processMS, "save", saveMS, "sum", sum)
}