1 /*
2 * Copyright (C) 2021 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 package com.google.android.renderscript
18
19 import android.graphics.Bitmap
20 import java.lang.IllegalArgumentException
21
22 // This string is used for error messages.
23 private const val externalName = "RenderScript Toolkit"
24
25 /**
26 * A collection of high-performance graphic utility functions like blur and blend.
27 *
28 * This toolkit provides ten image manipulation functions: blend, blur, color matrix, convolve,
29 * histogram, histogramDot, lut, lut3d, resize, and YUV to RGB. These functions execute
30 * multithreaded on the CPU.
31 *
32 * Most of the functions have two variants: one that manipulates Bitmaps, the other ByteArrays.
33 * For ByteArrays, you need to specify the width and height of the data to be processed, as
34 * well as the number of bytes per pixel. For most use cases, this will be 4.
35 *
36 * The Toolkit creates a thread pool that's used for processing the functions. The threads live
37 * for the duration of the application. They can be destroyed by calling the method shutdown().
38 *
39 * This library is thread safe. You can call methods from different poolThreads. The functions will
40 * execute sequentially.
41 *
42 * A native C++ version of this Toolkit is available. Check the RenderScriptToolkit.h file in the
43 * cpp directory.
44 *
45 * This toolkit can be used as a replacement for most RenderScript Intrinsic functions. Compared
46 * to RenderScript, it's simpler to use and more than twice as fast on the CPU. However RenderScript
47 * Intrinsics allow more flexibility for the type of allocation supported. In particular, this
48 * toolkit does not support allocations of floats.
49 */
50 object Toolkit {
51 /**
52 * Blends a source buffer with the destination buffer.
53 *
54 * Blends a source buffer and a destination buffer, placing the result in the destination
55 * buffer. The blending is done pairwise between two corresponding RGBA values found in
56 * each buffer. The mode parameter specifies one of fifteen supported blending operations.
57 * See {@link BlendingMode}.
58 *
59 * A variant of this method is also available to blend Bitmaps.
60 *
61 * An optional range parameter can be set to restrict the operation to a rectangular subset
62 * of each buffer. If provided, the range must be wholly contained with the dimensions
63 * described by sizeX and sizeY.
64 *
65 * The source and destination buffer must have the same dimensions. Both arrays should have
66 * a size greater or equal to sizeX * sizeY * 4. The buffers have a row-major layout.
67 *
68 * @param mode The specific blending operation to do.
69 * @param sourceArray The RGBA input buffer.
70 * @param destArray The destination buffer. Used for input and output.
71 * @param sizeX The width of both buffers, as a number of RGBA values.
72 * @param sizeY The height of both buffers, as a number of RGBA values.
73 * @param restriction When not null, restricts the operation to a 2D range of pixels.
74 */
75 @JvmOverloads
blendnull76 fun blend(
77 mode: BlendingMode,
78 sourceArray: ByteArray,
79 destArray: ByteArray,
80 sizeX: Int,
81 sizeY: Int,
82 restriction: Range2d? = null
83 ) {
84 require(sourceArray.size >= sizeX * sizeY * 4) {
85 "$externalName blend. sourceArray is too small for the given dimensions. " +
86 "$sizeX*$sizeY*4 < ${sourceArray.size}."
87 }
88 require(destArray.size >= sizeX * sizeY * 4) {
89 "$externalName blend. sourceArray is too small for the given dimensions. " +
90 "$sizeX*$sizeY*4 < ${sourceArray.size}."
91 }
92 validateRestriction("blend", sizeX, sizeY, restriction)
93
94 nativeBlend(nativeHandle, mode.value, sourceArray, destArray, sizeX, sizeY, restriction)
95 }
96
97 /**
98 * Blends a source bitmap with the destination bitmap.
99 *
100 * Blends a source bitmap and a destination bitmap, placing the result in the destination
101 * bitmap. The blending is done pairwise between two corresponding RGBA values found in
102 * each bitmap. The mode parameter specify one of fifteen supported blending operations.
103 * See {@link BlendingMode}.
104 *
105 * A variant of this method is available to blend ByteArrays.
106 *
107 * The bitmaps should have identical width and height, and have a config of ARGB_8888.
108 * Bitmaps with a stride different than width * vectorSize are not currently supported.
109 *
110 * An optional range parameter can be set to restrict the operation to a rectangular subset
111 * of each bitmap. If provided, the range must be wholly contained with the dimensions
112 * of the bitmap.
113 *
114 * @param mode The specific blending operation to do.
115 * @param sourceBitmap The RGBA input buffer.
116 * @param destBitmap The destination buffer. Used for input and output.
117 * @param restriction When not null, restricts the operation to a 2D range of pixels.
118 */
119 @JvmOverloads
blendnull120 fun blend(
121 mode: BlendingMode,
122 sourceBitmap: Bitmap,
123 destBitmap: Bitmap,
124 restriction: Range2d? = null
125 ) {
126 validateBitmap("blend", sourceBitmap)
127 validateBitmap("blend", destBitmap)
128 require(
129 sourceBitmap.width == destBitmap.width &&
130 sourceBitmap.height == destBitmap.height
131 ) {
132 "$externalName blend. Source and destination bitmaps should be the same size. " +
133 "${sourceBitmap.width}x${sourceBitmap.height} and " +
134 "${destBitmap.width}x${destBitmap.height} provided."
135 }
136 require(sourceBitmap.config == destBitmap.config) {
137 "RenderScript Toolkit blend. Source and destination bitmaps should have the same " +
138 "config. ${sourceBitmap.config} and ${destBitmap.config} provided."
139 }
140 validateRestriction("blend", sourceBitmap.width, sourceBitmap.height, restriction)
141
142 nativeBlendBitmap(nativeHandle, mode.value, sourceBitmap, destBitmap, restriction)
143 }
144
145 /**
146 * Blurs an image.
147 *
148 * Performs a Gaussian blur of an image and returns result in a ByteArray buffer. A variant of
149 * this method is available to blur Bitmaps.
150 *
151 * The radius determines which pixels are used to compute each blurred pixels. This Toolkit
152 * accepts values between 1 and 25. Larger values create a more blurred effect but also
153 * take longer to compute. When the radius extends past the edge, the edge pixel will
154 * be used as replacement for the pixel that's out off boundary.
155 *
156 * Each input pixel can either be represented by four bytes (RGBA format) or one byte
157 * for the less common blurring of alpha channel only image.
158 *
159 * An optional range parameter can be set to restrict the operation to a rectangular subset
160 * of each buffer. If provided, the range must be wholly contained with the dimensions
161 * described by sizeX and sizeY. NOTE: The output buffer will still be full size, with the
162 * section that's not blurred all set to 0. This is to stay compatible with RenderScript.
163 *
164 * The source buffer should be large enough for sizeX * sizeY * mVectorSize bytes. It has a
165 * row-major layout.
166 *
167 * @param inputArray The buffer of the image to be blurred.
168 * @param vectorSize Either 1 or 4, the number of bytes in each cell, i.e. A vs. RGBA.
169 * @param sizeX The width of both buffers, as a number of 1 or 4 byte cells.
170 * @param sizeY The height of both buffers, as a number of 1 or 4 byte cells.
171 * @param radius The radius of the pixels used to blur, a value from 1 to 25.
172 * @param restriction When not null, restricts the operation to a 2D range of pixels.
173 * @return The blurred pixels, a ByteArray of size.
174 */
175 @JvmOverloads
blurnull176 fun blur(
177 inputArray: ByteArray,
178 vectorSize: Int,
179 sizeX: Int,
180 sizeY: Int,
181 radius: Int = 5,
182 restriction: Range2d? = null
183 ): ByteArray {
184 require(vectorSize == 1 || vectorSize == 4) {
185 "$externalName blur. The vectorSize should be 1 or 4. $vectorSize provided."
186 }
187 require(inputArray.size >= sizeX * sizeY * vectorSize) {
188 "$externalName blur. inputArray is too small for the given dimensions. " +
189 "$sizeX*$sizeY*$vectorSize < ${inputArray.size}."
190 }
191 require(radius in 1..25) {
192 "$externalName blur. The radius should be between 1 and 25. $radius provided."
193 }
194 validateRestriction("blur", sizeX, sizeY, restriction)
195
196 val outputArray = ByteArray(inputArray.size)
197 nativeBlur(
198 nativeHandle, inputArray, vectorSize, sizeX, sizeY, radius, outputArray, restriction
199 )
200 return outputArray
201 }
202
203 /**
204 * Blurs an image.
205 *
206 * Performs a Gaussian blur of a Bitmap and returns result as a Bitmap. A variant of
207 * this method is available to blur ByteArrays.
208 *
209 * The radius determines which pixels are used to compute each blurred pixels. This Toolkit
210 * accepts values between 1 and 25. Larger values create a more blurred effect but also
211 * take longer to compute. When the radius extends past the edge, the edge pixel will
212 * be used as replacement for the pixel that's out off boundary.
213 *
214 * This method supports input Bitmap of config ARGB_8888 and ALPHA_8. Bitmaps with a stride
215 * different than width * vectorSize are not currently supported. The returned Bitmap has the
216 * same config.
217 *
218 * An optional range parameter can be set to restrict the operation to a rectangular subset
219 * of each buffer. If provided, the range must be wholly contained with the dimensions
220 * described by sizeX and sizeY. NOTE: The output Bitmap will still be full size, with the
221 * section that's not blurred all set to 0. This is to stay compatible with RenderScript.
222 *
223 * @param inputBitmap The buffer of the image to be blurred.
224 * @param radius The radius of the pixels used to blur, a value from 1 to 25. Default is 5.
225 * @param restriction When not null, restricts the operation to a 2D range of pixels.
226 * @return The blurred Bitmap.
227 */
228 @JvmOverloads
blurnull229 fun blur(inputBitmap: Bitmap, radius: Int = 5, restriction: Range2d? = null): Bitmap {
230 validateBitmap("blur", inputBitmap)
231 require(radius in 1..25) {
232 "$externalName blur. The radius should be between 1 and 25. $radius provided."
233 }
234 validateRestriction("blur", inputBitmap.width, inputBitmap.height, restriction)
235
236 val outputBitmap = createCompatibleBitmap(inputBitmap)
237 nativeBlurBitmap(nativeHandle, inputBitmap, outputBitmap, radius, restriction)
238 return outputBitmap
239 }
240
241 /**
242 * Identity matrix that can be passed to the {@link RenderScriptToolkit::colorMatrix} method.
243 *
244 * Using this matrix will result in no change to the pixel through multiplication although
245 * the pixel value can still be modified by the add vector, or transformed to a different
246 * format.
247 */
248 val identityMatrix: FloatArray
249 get() = floatArrayOf(
250 1f, 0f, 0f, 0f,
251 0f, 1f, 0f, 0f,
252 0f, 0f, 1f, 0f,
253 0f, 0f, 0f, 1f
254 )
255
256 /**
257 * Matrix to turn color pixels to a grey scale.
258 *
259 * Use this matrix with the {@link RenderScriptToolkit::colorMatrix} method to convert an
260 * image from color to greyscale.
261 */
262 val greyScaleColorMatrix: FloatArray
263 get() = floatArrayOf(
264 0.299f, 0.299f, 0.299f, 0f,
265 0.587f, 0.587f, 0.587f, 0f,
266 0.114f, 0.114f, 0.114f, 0f,
267 0f, 0f, 0f, 1f
268 )
269
270 /**
271 * Matrix to convert RGB to YUV.
272 *
273 * Use this matrix with the {@link RenderScriptToolkit::colorMatrix} method to convert the
274 * first three bytes of each pixel from RGB to YUV. This leaves the last byte (the alpha
275 * channel) untouched.
276 *
277 * This is a simplistic conversion. Most YUV buffers have more complicated format, not supported
278 * by this method.
279 */
280 val rgbToYuvMatrix: FloatArray
281 get() = floatArrayOf(
282 0.299f, -0.14713f, 0.615f, 0f,
283 0.587f, -0.28886f, -0.51499f, 0f,
284 0.114f, 0.436f, -0.10001f, 0f,
285 0f, 0f, 0f, 1f
286 )
287
288 /**
289 * Matrix to convert YUV to RGB.
290 *
291 * Use this matrix with the {@link RenderScriptToolkit::colorMatrix} method to convert the
292 * first three bytes of each pixel from YUV to RGB. This leaves the last byte (the alpha
293 * channel) untouched.
294 *
295 * This is a simplistic conversion. Most YUV buffers have more complicated format, not supported
296 * by this method. Use {@link RenderScriptToolkit::yuvToRgb} to convert these buffers.
297 */
298 val yuvToRgbMatrix: FloatArray
299 get() = floatArrayOf(
300 1f, 1f, 1f, 0f,
301 0f, -0.39465f, 2.03211f, 0f,
302 1.13983f, -0.5806f, 0f, 0f,
303 0f, 0f, 0f, 1f
304 )
305
306 /**
307 * Transform an image using a color matrix.
308 *
309 * Converts a 2D array of vectors of unsigned bytes, multiplying each vectors by a 4x4 matrix
310 * and adding an optional vector.
311 *
312 * Each input vector is composed of 1-4 unsigned bytes. If less than 4 bytes, it's extended to
313 * 4, padding with zeroes. The unsigned bytes are converted from 0-255 to 0.0-1.0 floats
314 * before the multiplication is done.
315 *
316 * The resulting value is normalized from 0.0-1.0 to a 0-255 value and stored in the output.
317 * If the output vector size is less than four, the unused channels are discarded.
318 *
319 * If addVector is not specified, a vector of zeroes is added, i.e. a noop.
320 *
321 * Like the RenderScript Intrinsics, vectorSize of size 3 are padded to occupy 4 bytes.
322 *
323 * Check identityMatrix, greyScaleColorMatrix, rgbToYuvMatrix, and yuvToRgbMatrix for sample
324 * matrices. The YUV conversion may not work for all color spaces.
325 *
326 * @param inputArray The buffer of the image to be converted.
327 * @param inputVectorSize The number of bytes in each input cell, a value from 1 to 4.
328 * @param sizeX The width of both buffers, as a number of 1 to 4 byte cells.
329 * @param sizeY The height of both buffers, as a number of 1 to 4 byte cells.
330 * @param outputVectorSize The number of bytes in each output cell, a value from 1 to 4.
331 * @param matrix The 4x4 matrix to multiply, in row major format.
332 * @param addVector A vector of four floats that's added to the result of the multiplication.
333 * @param restriction When not null, restricts the operation to a 2D range of pixels.
334 * @return The converted buffer.
335 */
336 @JvmOverloads
colorMatrixnull337 fun colorMatrix(
338 inputArray: ByteArray,
339 inputVectorSize: Int,
340 sizeX: Int,
341 sizeY: Int,
342 outputVectorSize: Int,
343 matrix: FloatArray,
344 addVector: FloatArray = floatArrayOf(0f, 0f, 0f, 0f),
345 restriction: Range2d? = null
346 ): ByteArray {
347 require(inputVectorSize in 1..4) {
348 "$externalName colorMatrix. The inputVectorSize should be between 1 and 4. " +
349 "$inputVectorSize provided."
350 }
351 require(outputVectorSize in 1..4) {
352 "$externalName colorMatrix. The outputVectorSize should be between 1 and 4. " +
353 "$outputVectorSize provided."
354 }
355 require(inputArray.size >= sizeX * sizeY * inputVectorSize) {
356 "$externalName colorMatrix. inputArray is too small for the given dimensions. " +
357 "$sizeX*$sizeY*$inputVectorSize < ${inputArray.size}."
358 }
359 require(matrix.size == 16) {
360 "$externalName colorMatrix. matrix should have 16 entries. ${matrix.size} provided."
361 }
362 require(addVector.size == 4) {
363 "$externalName colorMatrix. addVector should have 4 entries. " +
364 "${addVector.size} provided."
365 }
366 validateRestriction("colorMatrix", sizeX, sizeY, restriction)
367
368 val outputArray = ByteArray(sizeX * sizeY * paddedSize(outputVectorSize))
369 nativeColorMatrix(
370 nativeHandle, inputArray, inputVectorSize, sizeX, sizeY, outputArray, outputVectorSize,
371 matrix, addVector, restriction
372 )
373 return outputArray
374 }
375
376 /**
377 * Transform an image using a color matrix.
378 *
379 * Converts a bitmap, multiplying each RGBA value by a 4x4 matrix and adding an optional vector.
380 * Each byte of the RGBA is converted from 0-255 to 0.0-1.0 floats before the multiplication
381 * is done.
382 *
383 * Bitmaps with a stride different than width * vectorSize are not currently supported.
384 *
385 * The resulting value is normalized from 0.0-1.0 to a 0-255 value and stored in the output.
386 *
387 * If addVector is not specified, a vector of zeroes is added, i.e. a noop.
388 *
389 * Check identityMatrix, greyScaleColorMatrix, rgbToYuvMatrix, and yuvToRgbMatrix for sample
390 * matrices. The YUV conversion may not work for all color spaces.
391 *
392 * @param inputBitmap The image to be converted.
393 * @param matrix The 4x4 matrix to multiply, in row major format.
394 * @param addVector A vector of four floats that's added to the result of the multiplication.
395 * @param restriction When not null, restricts the operation to a 2D range of pixels.
396 * @return The converted buffer.
397 */
398 @JvmOverloads
colorMatrixnull399 fun colorMatrix(
400 inputBitmap: Bitmap,
401 matrix: FloatArray,
402 addVector: FloatArray = floatArrayOf(0f, 0f, 0f, 0f),
403 restriction: Range2d? = null
404 ): Bitmap {
405 validateBitmap("colorMatrix", inputBitmap)
406 require(matrix.size == 16) {
407 "$externalName colorMatrix. matrix should have 16 entries. ${matrix.size} provided."
408 }
409 require(addVector.size == 4) {
410 "$externalName colorMatrix. addVector should have 4 entries."
411 }
412 validateRestriction("colorMatrix", inputBitmap.width, inputBitmap.height, restriction)
413
414 val outputBitmap = createCompatibleBitmap(inputBitmap)
415 nativeColorMatrixBitmap(
416 nativeHandle,
417 inputBitmap,
418 outputBitmap,
419 matrix,
420 addVector,
421 restriction
422 )
423 return outputBitmap
424 }
425
426 /**
427 * Convolve a ByteArray.
428 *
429 * Applies a 3x3 or 5x5 convolution to the input array using the provided coefficients.
430 * A variant of this method is available to convolve Bitmaps.
431 *
432 * For 3x3 convolutions, 9 coefficients must be provided. For 5x5, 25 coefficients are needed.
433 * The coefficients should be provided in row-major format.
434 *
435 * When the square extends past the edge, the edge values will be used as replacement for the
436 * values that's are off boundary.
437 *
438 * Each input cell can either be represented by one to four bytes. Each byte is multiplied
439 * and accumulated independently of the other bytes of the cell.
440 *
441 * An optional range parameter can be set to restrict the convolve operation to a rectangular
442 * subset of each buffer. If provided, the range must be wholly contained with the dimensions
443 * described by sizeX and sizeY. NOTE: The output buffer will still be full size, with the
444 * section that's not convolved all set to 0. This is to stay compatible with RenderScript.
445 *
446 * The source array should be large enough for sizeX * sizeY * vectorSize bytes. It has a
447 * row-major layout. The output array will have the same dimensions.
448 *
449 * Like the RenderScript Intrinsics, vectorSize of size 3 are padded to occupy 4 bytes.
450 *
451 * @param inputArray The buffer of the image to be blurred.
452 * @param vectorSize The number of bytes in each cell, a value from 1 to 4.
453 * @param sizeX The width of both buffers, as a number of 1 or 4 byte cells.
454 * @param sizeY The height of both buffers, as a number of 1 or 4 byte cells.
455 * @param coefficients A FloatArray of size 9 or 25, containing the multipliers.
456 * @param restriction When not null, restricts the operation to a 2D range of pixels.
457 * @return The convolved array.
458 */
459 @JvmOverloads
convolvenull460 fun convolve(
461 inputArray: ByteArray,
462 vectorSize: Int,
463 sizeX: Int,
464 sizeY: Int,
465 coefficients: FloatArray,
466 restriction: Range2d? = null
467 ): ByteArray {
468 require(vectorSize in 1..4) {
469 "$externalName convolve. The vectorSize should be between 1 and 4. " +
470 "$vectorSize provided."
471 }
472 require(inputArray.size >= sizeX * sizeY * vectorSize) {
473 "$externalName convolve. inputArray is too small for the given dimensions. " +
474 "$sizeX*$sizeY*$vectorSize < ${inputArray.size}."
475 }
476 require(coefficients.size == 9 || coefficients.size == 25) {
477 "$externalName convolve. Only 3x3 or 5x5 convolutions are supported. " +
478 "${coefficients.size} coefficients provided."
479 }
480 validateRestriction("convolve", sizeX, sizeY, restriction)
481
482 val outputArray = ByteArray(inputArray.size)
483 nativeConvolve(
484 nativeHandle,
485 inputArray,
486 vectorSize,
487 sizeX,
488 sizeY,
489 outputArray,
490 coefficients,
491 restriction
492 )
493 return outputArray
494 }
495
496 /**
497 * Convolve a Bitmap.
498 *
499 * Applies a 3x3 or 5x5 convolution to the input Bitmap using the provided coefficients.
500 * A variant of this method is available to convolve ByteArrays. Bitmaps with a stride different
501 * than width * vectorSize are not currently supported.
502 *
503 * For 3x3 convolutions, 9 coefficients must be provided. For 5x5, 25 coefficients are needed.
504 * The coefficients should be provided in row-major format.
505 *
506 * Each input cell can either be represented by one to four bytes. Each byte is multiplied
507 * and accumulated independently of the other bytes of the cell.
508 *
509 * An optional range parameter can be set to restrict the convolve operation to a rectangular
510 * subset of each buffer. If provided, the range must be wholly contained with the dimensions
511 * described by sizeX and sizeY. NOTE: The output Bitmap will still be full size, with the
512 * section that's not convolved all set to 0. This is to stay compatible with RenderScript.
513 *
514 * @param inputBitmap The image to be blurred.
515 * @param coefficients A FloatArray of size 9 or 25, containing the multipliers.
516 * @param restriction When not null, restricts the operation to a 2D range of pixels.
517 * @return The convolved Bitmap.
518 */
519 @JvmOverloads
convolvenull520 fun convolve(
521 inputBitmap: Bitmap,
522 coefficients: FloatArray,
523 restriction: Range2d? = null
524 ): Bitmap {
525 validateBitmap("convolve", inputBitmap)
526 require(coefficients.size == 9 || coefficients.size == 25) {
527 "$externalName convolve. Only 3x3 or 5x5 convolutions are supported. " +
528 "${coefficients.size} coefficients provided."
529 }
530 validateRestriction("convolve", inputBitmap, restriction)
531
532 val outputBitmap = createCompatibleBitmap(inputBitmap)
533 nativeConvolveBitmap(nativeHandle, inputBitmap, outputBitmap, coefficients, restriction)
534 return outputBitmap
535 }
536
537 /**
538 * Compute the histogram of an image.
539 *
540 * Tallies how many times each of the 256 possible values of a byte is found in the input.
541 * A variant of this method is available to do the histogram of a Bitmap.
542 *
543 * An input cell can be represented by one to four bytes. The tally is done independently
544 * for each of the bytes of the cell. Correspondingly, the returned IntArray will have
545 * 256 * vectorSize entries. The counts for value 0 are consecutive, followed by those for
546 * value 1, etc.
547 *
548 * An optional range parameter can be set to restrict the operation to a rectangular subset
549 * of each buffer. If provided, the range must be wholly contained with the dimensions
550 * described by sizeX and sizeY.
551 *
552 * The source buffer should be large enough for sizeX * sizeY * vectorSize bytes. It has a
553 * row-major layout.
554 *
555 * Like the RenderScript Intrinsics, vectorSize of size 3 are padded to occupy 4 bytes.
556 *
557 * @param inputArray The buffer of the image to be analyzed.
558 * @param vectorSize The number of bytes in each cell, a value from 1 to 4.
559 * @param sizeX The width of the input buffers, as a number of 1 to 4 byte cells.
560 * @param sizeY The height of the input buffers, as a number of 1 to 4 byte cells.
561 * @param restriction When not null, restricts the operation to a 2D range of pixels.
562 * @return The resulting array of counts.
563 */
564 @JvmOverloads
histogramnull565 fun histogram(
566 inputArray: ByteArray,
567 vectorSize: Int,
568 sizeX: Int,
569 sizeY: Int,
570 restriction: Range2d? = null
571 ): IntArray {
572 require(vectorSize in 1..4) {
573 "$externalName histogram. The vectorSize should be between 1 and 4. " +
574 "$vectorSize provided."
575 }
576 require(inputArray.size >= sizeX * sizeY * vectorSize) {
577 "$externalName histogram. inputArray is too small for the given dimensions. " +
578 "$sizeX*$sizeY*$vectorSize < ${inputArray.size}."
579 }
580 validateRestriction("histogram", sizeX, sizeY, restriction)
581
582 val outputArray = IntArray(256 * paddedSize(vectorSize))
583 nativeHistogram(
584 nativeHandle,
585 inputArray,
586 vectorSize,
587 sizeX,
588 sizeY,
589 outputArray,
590 restriction
591 )
592 return outputArray
593 }
594
595 /**
596 * Compute the histogram of an image.
597 *
598 * Tallies how many times each of the 256 possible values of a byte is found in the bitmap.
599 * This method supports Bitmaps of config ARGB_8888 and ALPHA_8.
600 *
601 * For ARGB_8888, the tally is done independently of the four bytes. Correspondingly, the
602 * returned IntArray will have 4 * 256 entries. The counts for value 0 are consecutive,
603 * followed by those for value 1, etc.
604 *
605 * For ALPHA_8, an IntArray of size 256 is returned.
606 *
607 * Bitmaps with a stride different than width * vectorSize are not currently supported.
608 *
609 * A variant of this method is available to do the histogram of a ByteArray.
610 *
611 * An optional range parameter can be set to restrict the operation to a rectangular subset
612 * of each buffer. If provided, the range must be wholly contained with the dimensions
613 * described by sizeX and sizeY.
614 *
615 * @param inputBitmap The bitmap to be analyzed.
616 * @param restriction When not null, restricts the operation to a 2D range of pixels.
617 * @return The resulting array of counts.
618 */
619 @JvmOverloads
histogramnull620 fun histogram(
621 inputBitmap: Bitmap,
622 restriction: Range2d? = null
623 ): IntArray {
624 validateBitmap("histogram", inputBitmap)
625 validateRestriction("histogram", inputBitmap, restriction)
626
627 val outputArray = IntArray(256 * vectorSize(inputBitmap))
628 nativeHistogramBitmap(nativeHandle, inputBitmap, outputArray, restriction)
629 return outputArray
630 }
631
632 /**
633 * Compute the histogram of the dot product of an image.
634 *
635 * This method supports cells of 1 to 4 bytes in length. For each cell of the array,
636 * the dot product of its bytes with the provided coefficients is computed. The resulting
637 * floating point value is converted to an unsigned byte and tallied in the histogram.
638 *
639 * If coefficients is null, the coefficients used for RGBA luminosity calculation will be used,
640 * i.e. the values [0.299f, 0.587f, 0.114f, 0.f].
641 *
642 * Each coefficients must be >= 0 and their sum must be 1.0 or less. There must be the same
643 * number of coefficients as vectorSize.
644 *
645 * A variant of this method is available to do the histogram of a Bitmap.
646 *
647 * An optional range parameter can be set to restrict the operation to a rectangular subset
648 * of each buffer. If provided, the range must be wholly contained with the dimensions
649 * described by sizeX and sizeY.
650 *
651 * The source buffer should be large enough for sizeX * sizeY * vectorSize bytes. The returned
652 * array will have 256 ints.
653 *
654 * Like the RenderScript Intrinsics, vectorSize of size 3 are padded to occupy 4 bytes.
655 *
656 * @param inputArray The buffer of the image to be analyzed.
657 * @param vectorSize The number of bytes in each cell, a value from 1 to 4.
658 * @param sizeX The width of the input buffers, as a number of 1 to 4 byte cells.
659 * @param sizeY The height of the input buffers, as a number of 1 to 4 byte cells.
660 * @param coefficients The dot product multipliers. Size should equal vectorSize. Can be null.
661 * @param restriction When not null, restricts the operation to a 2D range of pixels.
662 * @return The resulting vector of counts.
663 */
664 @JvmOverloads
histogramDotnull665 fun histogramDot(
666 inputArray: ByteArray,
667 vectorSize: Int,
668 sizeX: Int,
669 sizeY: Int,
670 coefficients: FloatArray? = null,
671 restriction: Range2d? = null
672 ): IntArray {
673 require(vectorSize in 1..4) {
674 "$externalName histogramDot. The vectorSize should be between 1 and 4. " +
675 "$vectorSize provided."
676 }
677 require(inputArray.size >= sizeX * sizeY * vectorSize) {
678 "$externalName histogramDot. inputArray is too small for the given dimensions. " +
679 "$sizeX*$sizeY*$vectorSize < ${inputArray.size}."
680 }
681 validateHistogramDotCoefficients(coefficients, vectorSize)
682 validateRestriction("histogramDot", sizeX, sizeY, restriction)
683
684 val outputArray = IntArray(256)
685 val actualCoefficients = coefficients ?: floatArrayOf(0.299f, 0.587f, 0.114f, 0f)
686 nativeHistogramDot(
687 nativeHandle,
688 inputArray,
689 vectorSize,
690 sizeX,
691 sizeY,
692 outputArray,
693 actualCoefficients,
694 restriction
695 )
696 return outputArray
697 }
698
699 /**
700 * Compute the histogram of the dot product of an image.
701 *
702 * This method supports Bitmaps of config ARGB_8888 and ALPHA_8. For each pixel of the bitmap,
703 * the dot product of its bytes with the provided coefficients is computed. The resulting
704 * floating point value is converted to an unsigned byte and tallied in the histogram.
705 *
706 * If coefficients is null, the coefficients used for RGBA luminosity calculation will be used,
707 * i.e. the values [0.299f, 0.587f, 0.114f, 0.f].
708 *
709 * Each coefficients must be >= 0 and their sum must be 1.0 or less. For ARGB_8888, four values
710 * must be provided; for ALPHA_8, one.
711 *
712 * Bitmaps with a stride different than width * vectorSize are not currently supported.
713 *
714 * A variant of this method is available to do the histogram of a ByteArray.
715 *
716 * An optional range parameter can be set to restrict the operation to a rectangular subset
717 * of each buffer. If provided, the range must be wholly contained with the dimensions
718 * described by sizeX and sizeY.
719 *
720 * The returned array will have 256 ints.
721 *
722 * @param inputBitmap The bitmap to be analyzed.
723 * @param coefficients The one or four values used for the dot product. Can be null.
724 * @param restriction When not null, restricts the operation to a 2D range of pixels.
725 * @return The resulting vector of counts.
726 */
727 @JvmOverloads
histogramDotnull728 fun histogramDot(
729 inputBitmap: Bitmap,
730 coefficients: FloatArray? = null,
731 restriction: Range2d? = null
732 ): IntArray {
733 validateBitmap("histogramDot", inputBitmap)
734 validateHistogramDotCoefficients(coefficients, vectorSize(inputBitmap))
735 validateRestriction("histogramDot", inputBitmap, restriction)
736
737 val outputArray = IntArray(256)
738 val actualCoefficients = coefficients ?: floatArrayOf(0.299f, 0.587f, 0.114f, 0f)
739 nativeHistogramDotBitmap(
740 nativeHandle, inputBitmap, outputArray, actualCoefficients, restriction
741 )
742 return outputArray
743 }
744
745 /**
746 * Transform an image using a look up table
747 *
748 * Transforms an image by using a per-channel lookup table. Each channel of the input has an
749 * independent lookup table. The tables are 256 entries in size and can cover the full value
750 * range of a byte.
751 *
752 * The input array should be in RGBA format, where four consecutive bytes form an cell.
753 * A variant of this method is available to transform a Bitmap.
754 *
755 * An optional range parameter can be set to restrict the operation to a rectangular subset
756 * of each buffer. If provided, the range must be wholly contained with the dimensions
757 * described by sizeX and sizeY. NOTE: The output Bitmap will still be full size, with the
758 * section that's not convolved all set to 0. This is to stay compatible with RenderScript.
759 *
760 * The source array should be large enough for sizeX * sizeY * vectorSize bytes. The returned
761 * ray has the same dimensions as the input. The arrays have a row-major layout.
762 *
763 * @param inputArray The buffer of the image to be transformed.
764 * @param sizeX The width of both buffers, as a number of 4 byte cells.
765 * @param sizeY The height of both buffers, as a number of 4 byte cells.
766 * @param table The four arrays of 256 values that's used to convert each channel.
767 * @param restriction When not null, restricts the operation to a 2D range of pixels.
768 * @return The transformed image.
769 */
770 @JvmOverloads
lutnull771 fun lut(
772 inputArray: ByteArray,
773 sizeX: Int,
774 sizeY: Int,
775 table: LookupTable,
776 restriction: Range2d? = null
777 ): ByteArray {
778 require(inputArray.size >= sizeX * sizeY * 4) {
779 "$externalName lut. inputArray is too small for the given dimensions. " +
780 "$sizeX*$sizeY*4 < ${inputArray.size}."
781 }
782 validateRestriction("lut", sizeX, sizeY, restriction)
783
784 val outputArray = ByteArray(inputArray.size)
785 nativeLut(
786 nativeHandle,
787 inputArray,
788 outputArray,
789 sizeX,
790 sizeY,
791 table.red,
792 table.green,
793 table.blue,
794 table.alpha,
795 restriction
796 )
797 return outputArray
798 }
799
800 /**
801 * Transform an image using a look up table
802 *
803 * Transforms an image by using a per-channel lookup table. Each channel of the input has an
804 * independent lookup table. The tables are 256 entries in size and can cover the full value
805 * range of a byte.
806 *
807 * The input Bitmap should be in config ARGB_8888. A variant of this method is available to
808 * transform a ByteArray. Bitmaps with a stride different than width * vectorSize are not
809 * currently supported.
810 *
811 * An optional range parameter can be set to restrict the operation to a rectangular subset
812 * of each buffer. If provided, the range must be wholly contained with the dimensions
813 * described by sizeX and sizeY. NOTE: The output Bitmap will still be full size, with the
814 * section that's not convolved all set to 0. This is to stay compatible with RenderScript.
815 *
816 * @param inputBitmap The buffer of the image to be transformed.
817 * @param table The four arrays of 256 values that's used to convert each channel.
818 * @param restriction When not null, restricts the operation to a 2D range of pixels.
819 * @return The transformed image.
820 */
821 @JvmOverloads
lutnull822 fun lut(
823 inputBitmap: Bitmap,
824 table: LookupTable,
825 restriction: Range2d? = null
826 ): Bitmap {
827 validateBitmap("lut", inputBitmap)
828 validateRestriction("lut", inputBitmap, restriction)
829
830 val outputBitmap = createCompatibleBitmap(inputBitmap)
831 nativeLutBitmap(
832 nativeHandle,
833 inputBitmap,
834 outputBitmap,
835 table.red,
836 table.green,
837 table.blue,
838 table.alpha,
839 restriction
840 )
841 return outputBitmap
842 }
843
844 /**
845 * Transform an image using a 3D look up table
846 *
847 * Transforms an image, converting RGB to RGBA by using a 3D lookup table. The incoming R, G,
848 * and B values are normalized to the dimensions of the provided 3D buffer. The eight nearest
849 * values in that 3D buffer are sampled and linearly interpolated. The resulting RGBA entry
850 * is returned in the output array.
851 *
852 * The input array should be in RGBA format, where four consecutive bytes form an cell.
853 * The fourth byte of each input cell is ignored. A variant of this method is also available
854 * to transform Bitmaps.
855 *
856 * An optional range parameter can be set to restrict the operation to a rectangular subset
857 * of each buffer. If provided, the range must be wholly contained with the dimensions
858 * described by sizeX and sizeY. NOTE: The output array will still be full size, with the
859 * section that's not convolved all set to 0. This is to stay compatible with RenderScript.
860 *
861 * The source array should be large enough for sizeX * sizeY * vectorSize bytes. The returned
862 * array will have the same dimensions. The arrays have a row-major layout.
863 *
864 * @param inputArray The buffer of the image to be transformed.
865 * @param sizeX The width of both buffers, as a number of 4 byte cells.
866 * @param sizeY The height of both buffers, as a number of 4 byte cells.
867 * @param cube The translation cube.
868 * @param restriction When not null, restricts the operation to a 2D range of pixels.
869 * @return The transformed image.
870 */
871 @JvmOverloads
lut3dnull872 fun lut3d(
873 inputArray: ByteArray,
874 sizeX: Int,
875 sizeY: Int,
876 cube: Rgba3dArray,
877 restriction: Range2d? = null
878 ): ByteArray {
879 require(inputArray.size >= sizeX * sizeY * 4) {
880 "$externalName lut3d. inputArray is too small for the given dimensions. " +
881 "$sizeX*$sizeY*4 < ${inputArray.size}."
882 }
883 require(
884 cube.sizeX >= 2 && cube.sizeY >= 2 && cube.sizeZ >= 2 &&
885 cube.sizeX <= 256 && cube.sizeY <= 256 && cube.sizeZ <= 256
886 ) {
887 "$externalName lut3d. The dimensions of the cube should be between 2 and 256. " +
888 "(${cube.sizeX}, ${cube.sizeY}, ${cube.sizeZ}) provided."
889 }
890 validateRestriction("lut3d", sizeX, sizeY, restriction)
891
892 val outputArray = ByteArray(inputArray.size)
893 nativeLut3d(
894 nativeHandle, inputArray, outputArray, sizeX, sizeY, cube.values, cube.sizeX,
895 cube.sizeY, cube.sizeZ, restriction
896 )
897 return outputArray
898 }
899
900 /**
901 * Transform an image using a 3D look up table
902 *
903 * Transforms an image, converting RGB to RGBA by using a 3D lookup table. The incoming R, G,
904 * and B values are normalized to the dimensions of the provided 3D buffer. The eight nearest
905 * values in that 3D buffer are sampled and linearly interpolated. The resulting RGBA entry
906 * is returned in the output array.
907 *
908 * The input bitmap should be in RGBA_8888 format. The A channel is preserved. A variant of this
909 * method is also available to transform ByteArray. Bitmaps with a stride different than
910 * width * vectorSize are not currently supported.
911 *
912 * An optional range parameter can be set to restrict the operation to a rectangular subset
913 * of each buffer. If provided, the range must be wholly contained with the dimensions
914 * described by sizeX and sizeY. NOTE: The output array will still be full size, with the
915 * section that's not convolved all set to 0. This is to stay compatible with RenderScript.
916 *
917 * The source array should be large enough for sizeX * sizeY * vectorSize bytes. The returned
918 * array will have the same dimensions. The arrays have a row-major layout.
919 *
920 * @param inputBitmap The image to be transformed.
921 * @param cube The translation cube.
922 * @param restriction When not null, restricts the operation to a 2D range of pixels.
923 * @return The transformed image.
924 */
925 @JvmOverloads
lut3dnull926 fun lut3d(
927 inputBitmap: Bitmap,
928 cube: Rgba3dArray,
929 restriction: Range2d? = null
930 ): Bitmap {
931 validateBitmap("lut3d", inputBitmap)
932 validateRestriction("lut3d", inputBitmap, restriction)
933
934 val outputBitmap = createCompatibleBitmap(inputBitmap)
935 nativeLut3dBitmap(
936 nativeHandle, inputBitmap, outputBitmap, cube.values, cube.sizeX,
937 cube.sizeY, cube.sizeZ, restriction
938 )
939 return outputBitmap
940 }
941
942 /**
943 * Resize an image.
944 *
945 * Resizes an image using bicubic interpolation.
946 *
947 * This method supports elements of 1 to 4 bytes in length. Each byte of the element is
948 * interpolated independently from the others.
949 *
950 * An optional range parameter can be set to restrict the operation to a rectangular subset
951 * of the output buffer. The corresponding scaled range of the input will be used. If provided,
952 * the range must be wholly contained with the dimensions described by outputSizeX and
953 * outputSizeY.
954 *
955 * The input and output arrays have a row-major layout. The input array should be
956 * large enough for sizeX * sizeY * vectorSize bytes.
957 *
958 * Like the RenderScript Intrinsics, vectorSize of size 3 are padded to occupy 4 bytes.
959 *
960 * @param inputArray The buffer of the image to be resized.
961 * @param vectorSize The number of bytes in each element of both buffers. A value from 1 to 4.
962 * @param inputSizeX The width of the input buffer, as a number of 1-4 byte elements.
963 * @param inputSizeY The height of the input buffer, as a number of 1-4 byte elements.
964 * @param outputSizeX The width of the output buffer, as a number of 1-4 byte elements.
965 * @param outputSizeY The height of the output buffer, as a number of 1-4 byte elements.
966 * @param restriction When not null, restricts the operation to a 2D range of pixels.
967 * @return An array that contains the rescaled image.
968 */
969 @JvmOverloads
resizenull970 fun resize(
971 inputArray: ByteArray,
972 vectorSize: Int,
973 inputSizeX: Int,
974 inputSizeY: Int,
975 outputSizeX: Int,
976 outputSizeY: Int,
977 restriction: Range2d? = null
978 ): ByteArray {
979 require(vectorSize in 1..4) {
980 "$externalName resize. The vectorSize should be between 1 and 4. $vectorSize provided."
981 }
982 require(inputArray.size >= inputSizeX * inputSizeY * vectorSize) {
983 "$externalName resize. inputArray is too small for the given dimensions. " +
984 "$inputSizeX*$inputSizeY*$vectorSize < ${inputArray.size}."
985 }
986 validateRestriction("resize", outputSizeX, outputSizeY, restriction)
987
988 val outputArray = ByteArray(outputSizeX * outputSizeY * paddedSize(vectorSize))
989 nativeResize(
990 nativeHandle,
991 inputArray,
992 vectorSize,
993 inputSizeX,
994 inputSizeY,
995 outputArray,
996 outputSizeX,
997 outputSizeY,
998 restriction
999 )
1000 return outputArray
1001 }
1002
1003 /**
1004 * Resize an image.
1005 *
1006 * Resizes an image using bicubic interpolation.
1007 *
1008 * This method supports input Bitmap of config ARGB_8888 and ALPHA_8. The returned Bitmap
1009 * has the same config. Bitmaps with a stride different than width * vectorSize are not
1010 * currently supported.
1011 *
1012 * An optional range parameter can be set to restrict the operation to a rectangular subset
1013 * of the output buffer. The corresponding scaled range of the input will be used. If provided,
1014 * the range must be wholly contained with the dimensions described by outputSizeX and
1015 * outputSizeY.
1016 *
1017 * @param inputBitmap The Bitmap to be resized.
1018 * @param outputSizeX The width of the output buffer, as a number of 1-4 byte elements.
1019 * @param outputSizeY The height of the output buffer, as a number of 1-4 byte elements.
1020 * @param restriction When not null, restricts the operation to a 2D range of pixels.
1021 * @return A Bitmap that contains the rescaled image.
1022 */
1023 @JvmOverloads
resizenull1024 fun resize(
1025 inputBitmap: Bitmap,
1026 outputSizeX: Int,
1027 outputSizeY: Int,
1028 restriction: Range2d? = null
1029 ): Bitmap {
1030 validateBitmap("resize", inputBitmap)
1031 validateRestriction("resize", outputSizeX, outputSizeY, restriction)
1032
1033 val outputBitmap = Bitmap.createBitmap(outputSizeX, outputSizeY, Bitmap.Config.ARGB_8888)
1034 nativeResizeBitmap(nativeHandle, inputBitmap, outputBitmap, restriction)
1035 return outputBitmap
1036 }
1037
1038 /**
1039 * Convert an image from YUV to RGB.
1040 *
1041 * Converts a YUV buffer to RGB. The input array should be supplied in a supported YUV format.
1042 * The output is RGBA; the alpha channel will be set to 255.
1043 *
1044 * Note that for YV12 and a sizeX that's not a multiple of 32, the RenderScript Intrinsic may
1045 * not have converted the image correctly. This Toolkit method should.
1046 *
1047 * @param inputArray The buffer of the image to be converted.
1048 * @param sizeX The width in pixels of the image.
1049 * @param sizeY The height in pixels of the image.
1050 * @param format Either YV12 or NV21.
1051 * @return The converted image as a byte array.
1052 */
yuvToRgbnull1053 fun yuvToRgb(inputArray: ByteArray, sizeX: Int, sizeY: Int, format: YuvFormat): ByteArray {
1054 require(sizeX % 2 == 0 && sizeY % 2 == 0) {
1055 "$externalName yuvToRgb. Non-even dimensions are not supported. " +
1056 "$sizeX and $sizeY were provided."
1057 }
1058
1059 val outputArray = ByteArray(sizeX * sizeY * 4)
1060 nativeYuvToRgb(nativeHandle, inputArray, outputArray, sizeX, sizeY, format.value)
1061 return outputArray
1062 }
1063
1064 /**
1065 * Convert an image from YUV to an RGB Bitmap.
1066 *
1067 * Converts a YUV buffer to an RGB Bitmap. The input array should be supplied in a supported
1068 * YUV format. The output is RGBA; the alpha channel will be set to 255.
1069 *
1070 * Note that for YV12 and a sizeX that's not a multiple of 32, the RenderScript Intrinsic may
1071 * not have converted the image correctly. This Toolkit method should.
1072 *
1073 * @param inputArray The buffer of the image to be converted.
1074 * @param sizeX The width in pixels of the image.
1075 * @param sizeY The height in pixels of the image.
1076 * @param format Either YV12 or NV21.
1077 * @return The converted image.
1078 */
yuvToRgbBitmapnull1079 fun yuvToRgbBitmap(inputArray: ByteArray, sizeX: Int, sizeY: Int, format: YuvFormat): Bitmap {
1080 require(sizeX % 2 == 0 && sizeY % 2 == 0) {
1081 "$externalName yuvToRgbBitmap. Non-even dimensions are not supported. " +
1082 "$sizeX and $sizeY were provided."
1083 }
1084
1085 val outputBitmap = Bitmap.createBitmap(sizeX, sizeY, Bitmap.Config.ARGB_8888)
1086 nativeYuvToRgbBitmap(nativeHandle, inputArray, sizeX, sizeY, outputBitmap, format.value)
1087 return outputBitmap
1088 }
1089
1090 init {
1091 System.loadLibrary("renderscript-toolkit")
1092 nativeHandle = createNative()
1093 }
1094
1095 /**
1096 * Shutdown the thread pool.
1097 *
1098 * Waits for the threads to complete their work and destroys them.
1099 *
1100 * An application should call this method only if it is sure that it won't call the
1101 * toolkit again, as it is irreversible.
1102 */
shutdownnull1103 fun shutdown() {
1104 destroyNative(nativeHandle)
1105 nativeHandle = 0
1106 }
1107
1108 private var nativeHandle: Long = 0
1109
createNativenull1110 private external fun createNative(): Long
1111
1112 private external fun destroyNative(nativeHandle: Long)
1113
1114 private external fun nativeBlend(
1115 nativeHandle: Long,
1116 mode: Int,
1117 sourceArray: ByteArray,
1118 destArray: ByteArray,
1119 sizeX: Int,
1120 sizeY: Int,
1121 restriction: Range2d?
1122 )
1123
1124 private external fun nativeBlendBitmap(
1125 nativeHandle: Long,
1126 mode: Int,
1127 sourceBitmap: Bitmap,
1128 destBitmap: Bitmap,
1129 restriction: Range2d?
1130 )
1131
1132 private external fun nativeBlur(
1133 nativeHandle: Long,
1134 inputArray: ByteArray,
1135 vectorSize: Int,
1136 sizeX: Int,
1137 sizeY: Int,
1138 radius: Int,
1139 outputArray: ByteArray,
1140 restriction: Range2d?
1141 )
1142
1143 private external fun nativeBlurBitmap(
1144 nativeHandle: Long,
1145 inputBitmap: Bitmap,
1146 outputBitmap: Bitmap,
1147 radius: Int,
1148 restriction: Range2d?
1149 )
1150
1151 private external fun nativeColorMatrix(
1152 nativeHandle: Long,
1153 inputArray: ByteArray,
1154 inputVectorSize: Int,
1155 sizeX: Int,
1156 sizeY: Int,
1157 outputArray: ByteArray,
1158 outputVectorSize: Int,
1159 matrix: FloatArray,
1160 addVector: FloatArray,
1161 restriction: Range2d?
1162 )
1163
1164 private external fun nativeColorMatrixBitmap(
1165 nativeHandle: Long,
1166 inputBitmap: Bitmap,
1167 outputBitmap: Bitmap,
1168 matrix: FloatArray,
1169 addVector: FloatArray,
1170 restriction: Range2d?
1171 )
1172
1173 private external fun nativeConvolve(
1174 nativeHandle: Long,
1175 inputArray: ByteArray,
1176 vectorSize: Int,
1177 sizeX: Int,
1178 sizeY: Int,
1179 outputArray: ByteArray,
1180 coefficients: FloatArray,
1181 restriction: Range2d?
1182 )
1183
1184 private external fun nativeConvolveBitmap(
1185 nativeHandle: Long,
1186 inputBitmap: Bitmap,
1187 outputBitmap: Bitmap,
1188 coefficients: FloatArray,
1189 restriction: Range2d?
1190 )
1191
1192 private external fun nativeHistogram(
1193 nativeHandle: Long,
1194 inputArray: ByteArray,
1195 vectorSize: Int,
1196 sizeX: Int,
1197 sizeY: Int,
1198 outputArray: IntArray,
1199 restriction: Range2d?
1200 )
1201
1202 private external fun nativeHistogramBitmap(
1203 nativeHandle: Long,
1204 inputBitmap: Bitmap,
1205 outputArray: IntArray,
1206 restriction: Range2d?
1207 )
1208
1209 private external fun nativeHistogramDot(
1210 nativeHandle: Long,
1211 inputArray: ByteArray,
1212 vectorSize: Int,
1213 sizeX: Int,
1214 sizeY: Int,
1215 outputArray: IntArray,
1216 coefficients: FloatArray,
1217 restriction: Range2d?
1218 )
1219
1220 private external fun nativeHistogramDotBitmap(
1221 nativeHandle: Long,
1222 inputBitmap: Bitmap,
1223 outputArray: IntArray,
1224 coefficients: FloatArray,
1225 restriction: Range2d?
1226 )
1227
1228 private external fun nativeLut(
1229 nativeHandle: Long,
1230 inputArray: ByteArray,
1231 outputArray: ByteArray,
1232 sizeX: Int,
1233 sizeY: Int,
1234 red: ByteArray,
1235 green: ByteArray,
1236 blue: ByteArray,
1237 alpha: ByteArray,
1238 restriction: Range2d?
1239 )
1240
1241 private external fun nativeLutBitmap(
1242 nativeHandle: Long,
1243 inputBitmap: Bitmap,
1244 outputBitmap: Bitmap,
1245 red: ByteArray,
1246 green: ByteArray,
1247 blue: ByteArray,
1248 alpha: ByteArray,
1249 restriction: Range2d?
1250 )
1251
1252 private external fun nativeLut3d(
1253 nativeHandle: Long,
1254 inputArray: ByteArray,
1255 outputArray: ByteArray,
1256 sizeX: Int,
1257 sizeY: Int,
1258 cube: ByteArray,
1259 cubeSizeX: Int,
1260 cubeSizeY: Int,
1261 cubeSizeZ: Int,
1262 restriction: Range2d?
1263 )
1264
1265 private external fun nativeLut3dBitmap(
1266 nativeHandle: Long,
1267 inputBitmap: Bitmap,
1268 outputBitmap: Bitmap,
1269 cube: ByteArray,
1270 cubeSizeX: Int,
1271 cubeSizeY: Int,
1272 cubeSizeZ: Int,
1273 restriction: Range2d?
1274 )
1275
1276 private external fun nativeResize(
1277 nativeHandle: Long,
1278 inputArray: ByteArray,
1279 vectorSize: Int,
1280 inputSizeX: Int,
1281 inputSizeY: Int,
1282 outputArray: ByteArray,
1283 outputSizeX: Int,
1284 outputSizeY: Int,
1285 restriction: Range2d?
1286 )
1287
1288 private external fun nativeResizeBitmap(
1289 nativeHandle: Long,
1290 inputBitmap: Bitmap,
1291 outputBitmap: Bitmap,
1292 restriction: Range2d?
1293 )
1294
1295 private external fun nativeYuvToRgb(
1296 nativeHandle: Long,
1297 inputArray: ByteArray,
1298 outputArray: ByteArray,
1299 sizeX: Int,
1300 sizeY: Int,
1301 format: Int
1302 )
1303
1304 private external fun nativeYuvToRgbBitmap(
1305 nativeHandle: Long,
1306 inputArray: ByteArray,
1307 sizeX: Int,
1308 sizeY: Int,
1309 outputBitmap: Bitmap,
1310 value: Int
1311 )
1312 }
1313
1314 /**
1315 * Determines how a source buffer is blended into a destination buffer.
1316 * See {@link RenderScriptToolkit::blend}.
1317 *
1318 * blend only works on 4 byte RGBA data. In the descriptions below, ".a" represents
1319 * the alpha channel.
1320 */
1321 enum class BlendingMode(val value: Int) {
1322 /**
1323 * dest = 0
1324 *
1325 * The destination is cleared, i.e. each pixel is set to (0, 0, 0, 0)
1326 */
1327 CLEAR(0),
1328
1329 /**
1330 * dest = src
1331 *
1332 * Sets each pixel of the destination to the corresponding one in the source.
1333 */
1334 SRC(1),
1335
1336 /**
1337 * dest = dest
1338 *
1339 * Leaves the destination untouched. This is a no-op.
1340 */
1341 DST(2),
1342
1343 /**
1344 * dest = src + dest * (1.0 - src.a)
1345 */
1346 SRC_OVER(3),
1347
1348 /**
1349 * dest = dest + src * (1.0 - dest.a)
1350 */
1351 DST_OVER(4),
1352
1353 /**
1354 * dest = src * dest.a
1355 */
1356 SRC_IN(5),
1357
1358 /**
1359 * dest = dest * src.a
1360 */
1361 DST_IN(6),
1362
1363 /**
1364 * dest = src * (1.0 - dest.a)
1365 */
1366 SRC_OUT(7),
1367
1368 /**
1369 * dest = dest * (1.0 - src.a)
1370 */
1371 DST_OUT(8),
1372
1373 /**
1374 * dest.rgb = src.rgb * dest.a + (1.0 - src.a) * dest.rgb, dest.a = dest.a
1375 */
1376 SRC_ATOP(9),
1377
1378 /**
1379 * dest = dest.rgb * src.a + (1.0 - dest.a) * src.rgb, dest.a = src.a
1380 */
1381 DST_ATOP(10),
1382
1383 /**
1384 * dest = {src.r ^ dest.r, src.g ^ dest.g, src.b ^ dest.b, src.a ^ dest.a}
1385 *
1386 * Note: this is NOT the Porter/Duff XOR mode; this is a bitwise xor.
1387 */
1388 XOR(11),
1389
1390 /**
1391 * dest = src * dest
1392 */
1393 MULTIPLY(12),
1394
1395 /**
1396 * dest = min(src + dest, 1.0)
1397 */
1398 ADD(13),
1399
1400 /**
1401 * dest = max(dest - src, 0.0)
1402 */
1403 SUBTRACT(14)
1404 }
1405
1406 /**
1407 * A translation table used by the lut method. For each potential red, green, blue, and alpha
1408 * value, specifies it's replacement value.
1409 *
1410 * The fields are initialized to be a no-op operation, i.e. replace 1 by 1, 2 by 2, etc.
1411 * You can modify just the values you're interested in having a translation.
1412 */
1413 class LookupTable {
<lambda>null1414 var red = ByteArray(256) { it.toByte() }
<lambda>null1415 var green = ByteArray(256) { it.toByte() }
<lambda>null1416 var blue = ByteArray(256) { it.toByte() }
<lambda>null1417 var alpha = ByteArray(256) { it.toByte() }
1418 }
1419
1420 /**
1421 * The YUV formats supported by yuvToRgb.
1422 */
1423 enum class YuvFormat(val value: Int) {
1424 NV21(0x11),
1425 YV12(0x32315659),
1426 }
1427
1428 /**
1429 * Define a range of data to process.
1430 *
1431 * This class is used to restrict a [Toolkit] operation to a rectangular subset of the input
1432 * tensor.
1433 *
1434 * @property startX The index of the first value to be included on the X axis.
1435 * @property endX The index after the last value to be included on the X axis.
1436 * @property startY The index of the first value to be included on the Y axis.
1437 * @property endY The index after the last value to be included on the Y axis.
1438 */
1439 data class Range2d(
1440 val startX: Int,
1441 val endX: Int,
1442 val startY: Int,
1443 val endY: Int
1444 ) {
1445 constructor() : this(0, 0, 0, 0)
1446 }
1447
1448 class Rgba3dArray(val values: ByteArray, val sizeX: Int, val sizeY: Int, val sizeZ: Int) {
1449 init {
1450 require(values.size >= sizeX * sizeY * sizeZ * 4)
1451 }
1452
getnull1453 operator fun get(x: Int, y: Int, z: Int): ByteArray {
1454 val index = indexOfVector(x, y, z)
1455 return ByteArray(4) { values[index + it] }
1456 }
1457
setnull1458 operator fun set(x: Int, y: Int, z: Int, value: ByteArray) {
1459 require(value.size == 4)
1460 val index = indexOfVector(x, y, z)
1461 for (i in 0..3) {
1462 values[index + i] = value[i]
1463 }
1464 }
1465
indexOfVectornull1466 private fun indexOfVector(x: Int, y: Int, z: Int): Int {
1467 require(x in 0 until sizeX)
1468 require(y in 0 until sizeY)
1469 require(z in 0 until sizeZ)
1470 return ((z * sizeY + y) * sizeX + x) * 4
1471 }
1472 }
1473
validateBitmapnull1474 internal fun validateBitmap(
1475 function: String,
1476 inputBitmap: Bitmap,
1477 alphaAllowed: Boolean = true
1478 ) {
1479 if (alphaAllowed) {
1480 require(
1481 inputBitmap.config == Bitmap.Config.ARGB_8888 ||
1482 inputBitmap.config == Bitmap.Config.ALPHA_8
1483 ) {
1484 "$externalName. $function supports only ARGB_8888 and ALPHA_8 bitmaps. " +
1485 "${inputBitmap.config} provided."
1486 }
1487 } else {
1488 require(inputBitmap.config == Bitmap.Config.ARGB_8888) {
1489 "$externalName. $function supports only ARGB_8888. " +
1490 "${inputBitmap.config} provided."
1491 }
1492 }
1493 require(inputBitmap.width * vectorSize(inputBitmap) == inputBitmap.rowBytes) {
1494 "$externalName $function. Only bitmaps with rowSize equal to the width * vectorSize are " +
1495 "currently supported. Provided were rowBytes=${inputBitmap.rowBytes}, " +
1496 "width={${inputBitmap.width}, and vectorSize=${vectorSize(inputBitmap)}."
1497 }
1498 }
1499
createCompatibleBitmapnull1500 internal fun createCompatibleBitmap(inputBitmap: Bitmap) =
1501 Bitmap.createBitmap(inputBitmap.width, inputBitmap.height, inputBitmap.config)
1502
1503 internal fun validateHistogramDotCoefficients(
1504 coefficients: FloatArray?,
1505 vectorSize: Int
1506 ) {
1507 require(coefficients == null || coefficients.size == vectorSize) {
1508 "$externalName histogramDot. The coefficients should be null or have $vectorSize values."
1509 }
1510 if (coefficients !== null) {
1511 var sum = 0f
1512 for (i in 0 until vectorSize) {
1513 require(coefficients[i] >= 0.0f) {
1514 "$externalName histogramDot. Coefficients should not be negative. " +
1515 "Coefficient $i was ${coefficients[i]}."
1516 }
1517 sum += coefficients[i]
1518 }
1519 require(sum <= 1.0f) {
1520 "$externalName histogramDot. Coefficients should add to 1 or less. Their sum is $sum."
1521 }
1522 }
1523 }
1524
validateRestrictionnull1525 internal fun validateRestriction(tag: String, bitmap: Bitmap, restriction: Range2d? = null) {
1526 validateRestriction(tag, bitmap.width, bitmap.height, restriction)
1527 }
1528
validateRestrictionnull1529 internal fun validateRestriction(
1530 tag: String,
1531 sizeX: Int,
1532 sizeY: Int,
1533 restriction: Range2d? = null
1534 ) {
1535 if (restriction == null) return
1536 require(restriction.startX < sizeX && restriction.endX <= sizeX) {
1537 "$externalName $tag. sizeX should be greater than restriction.startX and greater " +
1538 "or equal to restriction.endX. $sizeX, ${restriction.startX}, " +
1539 "and ${restriction.endX} were provided respectively."
1540 }
1541 require(restriction.startY < sizeY && restriction.endY <= sizeY) {
1542 "$externalName $tag. sizeY should be greater than restriction.startY and greater " +
1543 "or equal to restriction.endY. $sizeY, ${restriction.startY}, " +
1544 "and ${restriction.endY} were provided respectively."
1545 }
1546 require(restriction.startX < restriction.endX) {
1547 "$externalName $tag. Restriction startX should be less than endX. " +
1548 "${restriction.startX} and ${restriction.endX} were provided respectively."
1549 }
1550 require(restriction.startY < restriction.endY) {
1551 "$externalName $tag. Restriction startY should be less than endY. " +
1552 "${restriction.startY} and ${restriction.endY} were provided respectively."
1553 }
1554 }
1555
vectorSizenull1556 internal fun vectorSize(bitmap: Bitmap): Int {
1557 return when (bitmap.config) {
1558 Bitmap.Config.ARGB_8888 -> 4
1559 Bitmap.Config.ALPHA_8 -> 1
1560 else -> throw IllegalArgumentException(
1561 "$externalName. Only ARGB_8888 and ALPHA_8 Bitmap are supported."
1562 )
1563 }
1564 }
1565
paddedSizenull1566 internal fun paddedSize(vectorSize: Int) = if (vectorSize == 3) 4 else vectorSize
1567