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 *
31574936894d3044445a272b39f2d925af40ece5d8Ruben Brunk * @see CaptureResult#STATISTICS_LENS_SHADING_CORRECTION_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");
65574936894d3044445a272b39f2d925af40ece5d8Ruben Brunk        mColumns = checkArgumentPositive(columns, "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() +
70574936894d3044445a272b39f2d925af40ece5d8Ruben Brunk                    " length, received " + elements.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;
245574936894d3044445a272b39f2d925af40ece5d8Ruben Brunk}
246