-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhue-rotate-optimized.go
69 lines (59 loc) · 1.6 KB
/
hue-rotate-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
package filters
import (
"math"
"runtime"
"sync"
"time"
"go-image-processing/utilities"
)
const DEG float64 = math.Pi / 180
func HueRotate(path string, angle int) {
img, format, openMS, convertMS := open(path)
now := math.Round(float64(time.Now().UnixNano()) / 1000000)
cos := math.Cos(float64(angle) * DEG)
sin := math.Sin(float64(angle) * DEG)
matrix := [3]float64{
cos + (1-cos)/3,
(1-cos)/3 - math.Sqrt(float64(1)/3)*sin,
(1-cos)/3 + math.Sqrt(float64(1)/3)*sin,
}
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 {
r, g, b := img.Pix[i], img.Pix[i+1], img.Pix[i+2]
rr := utilities.MaxMin(
float64(r)*matrix[0]+float64(g)*matrix[1]+float64(b)*matrix[2],
255,
0,
)
rg := utilities.MaxMin(
float64(r)*matrix[2]+float64(g)*matrix[0]+float64(b)*matrix[1],
255,
0,
)
rb := utilities.MaxMin(
float64(r)*matrix[1]+float64(g)*matrix[2]+float64(b)*matrix[0],
255,
0,
)
img.Pix[i] = uint8(rr)
img.Pix[i+1] = uint8(rg)
img.Pix[i+2] = uint8(rb)
}
}
for t := 0; t < threads; t += 1 {
wg.Add(1)
go processing(t)
}
wg.Wait()
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)
}