convolution_test.go 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280
  1. package imaging
  2. import (
  3. "image"
  4. "image/color"
  5. "testing"
  6. )
  7. func TestConvolve3x3(t *testing.T) {
  8. testCases := []struct {
  9. desc string
  10. src image.Image
  11. kernel [9]float64
  12. options *ConvolveOptions
  13. want *image.NRGBA
  14. }{
  15. {
  16. "Convolve3x3 0x0",
  17. &image.NRGBA{
  18. Rect: image.Rect(0, 0, 0, 0),
  19. Stride: 0,
  20. Pix: []uint8{},
  21. },
  22. [9]float64{
  23. 0, 0, 0,
  24. 0, 1, 0,
  25. 0, 0, 0,
  26. },
  27. nil,
  28. &image.NRGBA{Rect: image.Rect(0, 0, 0, 0)},
  29. },
  30. {
  31. "Convolve3x3 4x4 identity",
  32. &image.NRGBA{
  33. Rect: image.Rect(-1, -1, 3, 3),
  34. Stride: 4 * 4,
  35. Pix: []uint8{
  36. 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f,
  37. 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f,
  38. 0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2a, 0x2b, 0x2c, 0x2d, 0x2e, 0x2f,
  39. 0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e, 0x3f,
  40. },
  41. },
  42. [9]float64{
  43. 0, 0, 0,
  44. 0, 1, 0,
  45. 0, 0, 0,
  46. },
  47. nil,
  48. &image.NRGBA{
  49. Rect: image.Rect(0, 0, 4, 4),
  50. Stride: 4 * 4,
  51. Pix: []uint8{
  52. 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f,
  53. 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f,
  54. 0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2a, 0x2b, 0x2c, 0x2d, 0x2e, 0x2f,
  55. 0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e, 0x3f,
  56. },
  57. },
  58. },
  59. {
  60. "Convolve3x3 4x4 abs",
  61. &image.NRGBA{
  62. Rect: image.Rect(-1, -1, 3, 3),
  63. Stride: 4 * 4,
  64. Pix: []uint8{
  65. 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f,
  66. 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f,
  67. 0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2a, 0x2b, 0x2c, 0x2d, 0x2e, 0x2f,
  68. 0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e, 0x3f,
  69. },
  70. },
  71. [9]float64{
  72. 0, 0, 0,
  73. 0, -1, 0,
  74. 0, 0, 0,
  75. },
  76. &ConvolveOptions{Abs: true},
  77. &image.NRGBA{
  78. Rect: image.Rect(0, 0, 4, 4),
  79. Stride: 4 * 4,
  80. Pix: []uint8{
  81. 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f,
  82. 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f,
  83. 0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2a, 0x2b, 0x2c, 0x2d, 0x2e, 0x2f,
  84. 0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e, 0x3f,
  85. },
  86. },
  87. },
  88. {
  89. "Convolve3x3 4x4 bias",
  90. &image.NRGBA{
  91. Rect: image.Rect(-1, -1, 3, 3),
  92. Stride: 4 * 4,
  93. Pix: []uint8{
  94. 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f,
  95. 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f,
  96. 0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2a, 0x2b, 0x2c, 0x2d, 0x2e, 0x2f,
  97. 0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e, 0x3f,
  98. },
  99. },
  100. [9]float64{
  101. 0, 0, 0,
  102. 0, 1, 0,
  103. 0, 0, 0,
  104. },
  105. &ConvolveOptions{Bias: 0x10},
  106. &image.NRGBA{
  107. Rect: image.Rect(0, 0, 4, 4),
  108. Stride: 4 * 4,
  109. Pix: []uint8{
  110. 0x10, 0x11, 0x12, 0x03, 0x14, 0x15, 0x16, 0x07, 0x18, 0x19, 0x1a, 0x0b, 0x1c, 0x1d, 0x1e, 0x0f,
  111. 0x20, 0x21, 0x22, 0x13, 0x24, 0x25, 0x26, 0x17, 0x28, 0x29, 0x2a, 0x1b, 0x2c, 0x2d, 0x2e, 0x1f,
  112. 0x30, 0x31, 0x32, 0x23, 0x34, 0x35, 0x36, 0x27, 0x38, 0x39, 0x3a, 0x2b, 0x3c, 0x3d, 0x3e, 0x2f,
  113. 0x40, 0x41, 0x42, 0x33, 0x44, 0x45, 0x46, 0x37, 0x48, 0x49, 0x4a, 0x3b, 0x4c, 0x4d, 0x4e, 0x3f,
  114. },
  115. },
  116. },
  117. {
  118. "Convolve3x3 4x4 norm",
  119. &image.NRGBA{
  120. Rect: image.Rect(-1, -1, 3, 3),
  121. Stride: 4 * 4,
  122. Pix: []uint8{
  123. 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f,
  124. 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f,
  125. 0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2a, 0x2b, 0x2c, 0x2d, 0x2e, 0x2f,
  126. 0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e, 0x3f,
  127. },
  128. },
  129. [9]float64{
  130. 1, 1, 1,
  131. 1, 1, 1,
  132. 1, 1, 1,
  133. },
  134. &ConvolveOptions{Normalize: true},
  135. &image.NRGBA{
  136. Rect: image.Rect(0, 0, 4, 4),
  137. Stride: 4 * 4,
  138. Pix: []uint8{
  139. 0x07, 0x08, 0x09, 0x03, 0x09, 0x0a, 0x0b, 0x07, 0x0d, 0x0e, 0x0f, 0x0b, 0x10, 0x11, 0x12, 0x0f,
  140. 0x11, 0x12, 0x13, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1a, 0x1b, 0x1b, 0x1c, 0x1d, 0x1f,
  141. 0x21, 0x22, 0x23, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2a, 0x2b, 0x2b, 0x2c, 0x2d, 0x2f,
  142. 0x2c, 0x2d, 0x2e, 0x33, 0x2f, 0x30, 0x31, 0x37, 0x33, 0x34, 0x35, 0x3b, 0x35, 0x36, 0x37, 0x3f,
  143. },
  144. },
  145. },
  146. {
  147. "Convolve3x3 3x3 laplacian",
  148. &image.NRGBA{
  149. Rect: image.Rect(-1, -1, 2, 2),
  150. Stride: 3 * 4,
  151. Pix: []uint8{
  152. 0x00, 0x01, 0x01, 0xff, 0x00, 0x01, 0x02, 0xff, 0x00, 0x01, 0x03, 0xff,
  153. 0x00, 0x01, 0x04, 0xff, 0x10, 0x10, 0x10, 0xff, 0x00, 0x01, 0x05, 0xff,
  154. 0x00, 0x01, 0x06, 0xff, 0x00, 0x01, 0x07, 0xff, 0x00, 0x01, 0x08, 0xff,
  155. },
  156. },
  157. [9]float64{
  158. -1, -1, -1,
  159. -1, 8, -1,
  160. -1, -1, -1,
  161. },
  162. nil,
  163. &image.NRGBA{
  164. Rect: image.Rect(0, 0, 3, 3),
  165. Stride: 3 * 4,
  166. Pix: []uint8{
  167. 0x00, 0x00, 0x00, 0xff, 0x00, 0x00, 0x00, 0xff, 0x00, 0x00, 0x00, 0xff,
  168. 0x00, 0x00, 0x00, 0xff, 0x80, 0x78, 0x5c, 0xff, 0x00, 0x00, 0x00, 0xff,
  169. 0x00, 0x00, 0x00, 0xff, 0x00, 0x00, 0x00, 0xff, 0x00, 0x00, 0x00, 0xff,
  170. },
  171. },
  172. },
  173. }
  174. for _, tc := range testCases {
  175. got := Convolve3x3(tc.src, tc.kernel, tc.options)
  176. want := tc.want
  177. if !compareNRGBA(got, want, 0) {
  178. t.Errorf("test [%s] failed: want %#v got %#v", tc.desc, want, got)
  179. }
  180. }
  181. }
  182. func TestConvolve5x5(t *testing.T) {
  183. testCases := []struct {
  184. desc string
  185. src image.Image
  186. kernel [25]float64
  187. options *ConvolveOptions
  188. want *image.NRGBA
  189. }{
  190. {
  191. "Convolve5x5 4x4 translate",
  192. &image.NRGBA{
  193. Rect: image.Rect(-1, -1, 3, 3),
  194. Stride: 4 * 4,
  195. Pix: []uint8{
  196. 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f,
  197. 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f,
  198. 0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2a, 0x2b, 0x2c, 0x2d, 0x2e, 0x2f,
  199. 0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e, 0x3f,
  200. },
  201. },
  202. [25]float64{
  203. 0, 0, 0, 0, 0,
  204. 0, 0, 0, 0, 0,
  205. 0, 0, 0, 0, 0,
  206. 0, 0, 0, 0, 0,
  207. 0, 0, 0, 0, 1,
  208. },
  209. nil,
  210. &image.NRGBA{
  211. Rect: image.Rect(0, 0, 4, 4),
  212. Stride: 4 * 4,
  213. Pix: []uint8{
  214. 0x28, 0x29, 0x2a, 0x03, 0x2c, 0x2d, 0x2e, 0x07, 0x2c, 0x2d, 0x2e, 0x0b, 0x2c, 0x2d, 0x2e, 0x0f,
  215. 0x38, 0x39, 0x3a, 0x13, 0x3c, 0x3d, 0x3e, 0x17, 0x3c, 0x3d, 0x3e, 0x1b, 0x3c, 0x3d, 0x3e, 0x1f,
  216. 0x38, 0x39, 0x3a, 0x23, 0x3c, 0x3d, 0x3e, 0x27, 0x3c, 0x3d, 0x3e, 0x2b, 0x3c, 0x3d, 0x3e, 0x2f,
  217. 0x38, 0x39, 0x3a, 0x33, 0x3c, 0x3d, 0x3e, 0x37, 0x3c, 0x3d, 0x3e, 0x3b, 0x3c, 0x3d, 0x3e, 0x3f,
  218. },
  219. },
  220. },
  221. }
  222. for _, tc := range testCases {
  223. got := Convolve5x5(tc.src, tc.kernel, tc.options)
  224. want := tc.want
  225. if !compareNRGBA(got, want, 0) {
  226. t.Errorf("test [%s] failed: want %#v got %#v", tc.desc, want, got)
  227. }
  228. }
  229. }
  230. func BenchmarkConvolve3x3(b *testing.B) {
  231. r := image.Rect(0, 0, 500, 500)
  232. img := image.NewRGBA(r)
  233. for x := r.Min.X; x < r.Max.X; x++ {
  234. for y := r.Min.Y; y < r.Max.Y; y++ {
  235. img.SetRGBA(x, y, color.RGBA{uint8(x % 255), uint8(y % 255), uint8((x + y) % 255), 255})
  236. }
  237. }
  238. b.ResetTimer()
  239. for i := 0; i < b.N; i++ {
  240. Convolve3x3(
  241. img,
  242. [9]float64{
  243. -1, -1, 0,
  244. -1, 0, 1,
  245. 0, 1, 1,
  246. },
  247. nil,
  248. )
  249. }
  250. }
  251. func BenchmarkConvolve5x5(b *testing.B) {
  252. r := image.Rect(0, 0, 500, 500)
  253. img := image.NewRGBA(r)
  254. for x := r.Min.X; x < r.Max.X; x++ {
  255. for y := r.Min.Y; y < r.Max.Y; y++ {
  256. img.SetRGBA(x, y, color.RGBA{uint8(x % 255), uint8(y % 255), uint8((x + y) % 255), 255})
  257. }
  258. }
  259. b.ResetTimer()
  260. for i := 0; i < b.N; i++ {
  261. Convolve5x5(
  262. img,
  263. [25]float64{
  264. -1, -1, -1, -1, 0,
  265. -1, -1, -1, 0, 1,
  266. -1, -1, 0, 1, 1,
  267. -1, 0, 1, 1, 1,
  268. 0, 1, 1, 1, 1,
  269. },
  270. nil,
  271. )
  272. }
  273. }