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