LensShadingMap.java revision 3c40a046cf0ea7b6af01ec93e5276eccb3234bfe
194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin/* 294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * Copyright (C) 2014 The Android Open Source Project 394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * Licensed under the Apache License, Version 2.0 (the "License"); 594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * you may not use this file except in compliance with the License. 694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * You may obtain a copy of the License at 794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * http://www.apache.org/licenses/LICENSE-2.0 994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 1094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * Unless required by applicable law or agreed to in writing, software 1194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * distributed under the License is distributed on an "AS IS" BASIS, 1294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 1394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * See the License for the specific language governing permissions and 1494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * limitations under the License. 1594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin */ 1694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 1794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkinpackage android.hardware.camera2; 1894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 1994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkinimport static com.android.internal.util.Preconditions.*; 2094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkinimport static android.hardware.camera2.RggbChannelVector.*; 2194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 223c40a046cf0ea7b6af01ec93e5276eccb3234bfeIgor Murashkinimport android.hardware.camera2.utils.HashCodeHelpers; 2394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 2494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkinimport java.util.Arrays; 2594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 2694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin/** 2794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * Immutable class for describing a {@code 4 x N x M} lens shading map of floats. 2894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 2994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @see CameraCharacteristics#LENS_SHADING_MAP 3094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin */ 3194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkinpublic final class LensShadingMap { 3294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 3394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin /** 3494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * The smallest gain factor in this map. 3594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 3694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * <p>All values in this map will be at least this large.</p> 3794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin */ 3894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin public static final float MINIMUM_GAIN_FACTOR = 1.0f; 3994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 4094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin /** 4194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * Create a new immutable LensShadingMap instance. 4294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 4394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * <p>The elements must be stored in a row-major order (fully packed).</p> 4494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 4594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * <p>This constructor takes over the array; do not write to the array afterwards.</p> 4694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 4794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @param elements 4894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * An array of elements whose length is 4994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * {@code RggbChannelVector.COUNT * rows * columns} 5094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 5194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @throws IllegalArgumentException 5294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * if the {@code elements} array length is invalid, 5394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * if any of the subelems are not finite or less than {@value #MINIMUM_GAIN_FACTOR}, 5494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * or if rows or columns is not positive 5594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @throws NullPointerException 5694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * if {@code elements} is {@code null} 5794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 5894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @hide 5994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin */ 6094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin public LensShadingMap(final float[] elements, final int rows, final int columns) { 6194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 6294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin mRows = checkArgumentPositive(rows, "rows must be positive"); 6394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin mColumns = checkArgumentPositive(rows, "columns must be positive"); 6494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin mElements = checkNotNull(elements, "elements must not be null"); 6594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 6694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin if (elements.length != getGainFactorCount()) { 6794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin throw new IllegalArgumentException("elements must be " + getGainFactorCount() + 6894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin " length"); 6994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin } 7094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 7194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin // Every element must be finite and >= 1.0f 7294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin checkArrayElementsInRange(elements, MINIMUM_GAIN_FACTOR, Float.MAX_VALUE, "elements"); 7394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin } 7494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 7594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin /** 7694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * Get the number of rows in this map. 7794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin */ 7894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin public int getRowCount() { 7994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin return mRows; 8094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin } 8194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 8294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin /** 8394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * Get the number of columns in this map. 8494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin */ 8594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin public int getColumnCount() { 8694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin return mColumns; 8794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin } 8894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 8994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin /** 9094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * Get the total number of gain factors in this map. 9194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 9294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * <p>A single gain factor contains exactly one color channel. 9394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * Use with {@link #copyGainFactors} to allocate a large-enough array.</p> 9494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin */ 9594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin public int getGainFactorCount() { 9694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin return mRows * mColumns * COUNT; 9794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin } 9894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 9994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin /** 10094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * Get a single color channel gain factor from this lens shading map by its row and column. 10194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 10294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * <p>The rows must be within the range [0, {@link #getRowCount}), 10394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * the column must be within the range [0, {@link #getColumnCount}), 10494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * and the color channel must be within the range [0, {@value RggbChannelVector#COUNT}).</p> 10594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 10694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * <p>The channel order is {@code [R, Geven, Godd, B]}, where 10794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * {@code Geven} is the green channel for the even rows of a Bayer pattern, and 10894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * {@code Godd} is the odd rows. 10994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * </p> 11094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 11194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @param colorChannel color channel from {@code [R, Geven, Godd, B]} 11294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @param column within the range [0, {@link #getColumnCount}) 11394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @param row within the range [0, {@link #getRowCount}) 11494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 11594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @return a gain factor >= {@value #MINIMUM_GAIN_FACTOR} 11694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 11794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @throws IllegalArgumentException if any of the parameters was out of range 11894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 11994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @see #RED 12094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @see #GREEN_EVEN 12194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @see #GREEN_ODD 12294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @see #BLUE 12394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @see #getRowCount 12494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @see #getColumnCount 12594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin */ 12694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin public float getGainFactor(final int colorChannel, final int column, final int row) { 12794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin if (colorChannel < 0 || colorChannel > COUNT) { 12894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin throw new IllegalArgumentException("colorChannel out of range"); 12994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin } else if (column < 0 || column >= mColumns) { 13094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin throw new IllegalArgumentException("column out of range"); 13194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin } else if (row < 0 || row >= mRows) { 13294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin throw new IllegalArgumentException("row out of range"); 13394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin } 13494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 13594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin return mElements[colorChannel + (row * mColumns + column) * COUNT ]; 13694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin } 13794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 13894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin /** 13994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * Get a gain factor vector from this lens shading map by its row and column. 14094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 14194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * <p>The rows must be within the range [0, {@link #getRowCount}), 14294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * the column must be within the range [0, {@link #getColumnCount}).</p> 14394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 14494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @param column within the range [0, {@link #getColumnCount}) 14594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @param row within the range [0, {@link #getRowCount}) 14694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 14794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @return an {@link RggbChannelVector} where each gain factor >= {@value #MINIMUM_GAIN_FACTOR} 14894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 14994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @throws IllegalArgumentException if any of the parameters was out of range 15094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 15194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @see #getRowCount 15294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @see #getColumnCount 15394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin */ 15494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin public RggbChannelVector getGainFactorVector(final int column, final int row) { 15594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin if (column < 0 || column >= mColumns) { 15694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin throw new IllegalArgumentException("column out of range"); 15794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin } else if (row < 0 || row >= mRows) { 15894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin throw new IllegalArgumentException("row out of range"); 15994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin } 16094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 16194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin final int offset = (row * mColumns + column) * COUNT; 16294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 16394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin final float red = 16494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin mElements[RED + offset]; 16594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin final float greenEven = 16694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin mElements[GREEN_EVEN + offset]; 16794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin final float greenOdd = 16894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin mElements[GREEN_ODD + offset]; 16994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin final float blue = 17094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin mElements[BLUE + offset]; 17194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 17294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin return new RggbChannelVector(red, greenEven, greenOdd, blue); 17394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin } 17494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 17594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin /** 17694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * Copy all gain factors in row-major order from this lens shading map into the destination. 17794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 17894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * <p>Each gain factor will be >= {@link #MINIMUM_GAIN_FACTOR}.</p> 17994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 18094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @param destination 18194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * an array big enough to hold at least {@link RggbChannelVector#COUNT} 18294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * elements after the {@code offset} 18394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @param offset 18494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * a non-negative offset into the array 18594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @throws NullPointerException 18694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * If {@code destination} was {@code null} 18794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @throws IllegalArgumentException 18894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * If offset was negative 18994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @throws ArrayIndexOutOfBoundsException 19094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * If there's not enough room to write the elements at the specified destination and 19194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * offset. 19294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 19394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @see CaptureResult#STATISTICS_LENS_SHADING_MAP 19494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin */ 19594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin public void copyGainFactors(final float[] destination, final int offset) { 19694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin checkArgumentNonnegative(offset, "offset must not be negative"); 19794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin checkNotNull(destination, "destination must not be null"); 19894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin if (destination.length + offset < getGainFactorCount()) { 19994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin throw new ArrayIndexOutOfBoundsException("destination too small to fit elements"); 20094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin } 20194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 20294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin System.arraycopy(mElements, /*srcPos*/0, destination, offset, getGainFactorCount()); 20394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin } 20494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 20594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin /** 20694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * Check if this LensShadingMap is equal to another LensShadingMap. 20794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 20894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * <p>Two lens shading maps are equal if and only if they have the same rows/columns, 20994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * and all of their elements are {@link Object#equals equal}.</p> 21094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * 21194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * @return {@code true} if the objects were equal, {@code false} otherwise 21294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin */ 21394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin @Override 21494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin public boolean equals(final Object obj) { 21594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin if (obj == null) { 21694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin return false; 21794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin } 21894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin if (this == obj) { 21994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin return true; 22094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin } 22194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin if (obj instanceof LensShadingMap) { 22294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin final LensShadingMap other = (LensShadingMap) obj; 22394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin return mRows == other.mRows 22494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin && mColumns == other.mColumns 22594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin && Arrays.equals(mElements, other.mElements); 22694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin } 22794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin return false; 22894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin } 22994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 23094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin /** 23194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin * {@inheritDoc} 23294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin */ 23394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin @Override 23494814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin public int hashCode() { 23594814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin int elemsHash = HashCodeHelpers.hashCode(mElements); 23694814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin return HashCodeHelpers.hashCode(mRows, mColumns, elemsHash); 23794814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin } 23894814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 23994814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin 24094814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin private final int mRows; 24194814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin private final int mColumns; 24294814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin private final float[] mElements; 24394814218d2313a989a5a8969f633e3fc33e43071Igor Murashkin}; 244