org.esa.s1tbx.fex.gpf.oceantools.WindFieldEstimationOp.java Source code

Java tutorial

Introduction

Here is the source code for org.esa.s1tbx.fex.gpf.oceantools.WindFieldEstimationOp.java

Source

/*
 * Copyright (C) 2015 by Array Systems Computing Inc. http://www.array.ca
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License as published by the Free
 * Software Foundation; either version 3 of the License, or (at your option)
 * any later version.
 * This program is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
 * more details.
 *
 * You should have received a copy of the GNU General Public License along
 * with this program; if not, see http://www.gnu.org/licenses/
 */
package org.esa.s1tbx.fex.gpf.oceantools;

import Jama.Matrix;
import com.bc.ceres.core.ProgressMonitor;
import edu.emory.mathcs.jtransforms.fft.DoubleFFT_1D;
import org.apache.commons.math3.util.FastMath;
import org.esa.snap.core.datamodel.Band;
import org.esa.snap.core.datamodel.MetadataElement;
import org.esa.snap.core.datamodel.Product;
import org.esa.snap.core.datamodel.TiePointGrid;
import org.esa.snap.core.dataop.downloadable.XMLSupport;
import org.esa.snap.core.gpf.Operator;
import org.esa.snap.core.gpf.OperatorException;
import org.esa.snap.core.gpf.OperatorSpi;
import org.esa.snap.core.gpf.Tile;
import org.esa.snap.core.gpf.annotations.OperatorMetadata;
import org.esa.snap.core.gpf.annotations.Parameter;
import org.esa.snap.core.gpf.annotations.SourceProduct;
import org.esa.snap.core.gpf.annotations.TargetProduct;
import org.esa.snap.core.util.ProductUtils;
import org.esa.snap.engine_utilities.datamodel.AbstractMetadata;
import org.esa.snap.engine_utilities.datamodel.Unit;
import org.esa.snap.engine_utilities.eo.Constants;
import org.esa.snap.engine_utilities.gpf.InputProductValidator;
import org.esa.snap.engine_utilities.gpf.OperatorUtils;
import org.esa.snap.engine_utilities.util.ResourceUtils;
import org.jdom2.Document;
import org.jdom2.Element;

import javax.media.jai.JAI;
import javax.media.jai.PlanarImage;
import javax.media.jai.RasterFactory;
import javax.media.jai.operator.MedianFilterDescriptor;
import javax.media.jai.operator.MedianFilterShape;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.awt.image.ColorModel;
import java.awt.image.DataBuffer;
import java.awt.image.DataBufferDouble;
import java.awt.image.Raster;
import java.awt.image.RenderedImage;
import java.awt.image.SampleModel;
import java.awt.image.WritableRaster;
import java.awt.image.renderable.ParameterBlock;
import java.io.File;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Hashtable;
import java.util.List;

/**
 * The wind field retrieval operator.
 * <p/>
 * The operator retrieves wind speed and direction from C-band SAR imagery. The wind direction is estimated
 * from the wind roll using a frequency domain method and the wind speed is estimated by using CMOD5 model
 * for the Normalized Radar Cross Section (NRCS).
 * <p/>
 * This operator supports only ERS and ENVISAT products. It is asssumed that the product has been calibrated
 * before applying this operator.
 * <p/>
 * [1] H. Hersbach, CMOD5, "An Improved Geophysical Model Function for ERS C-Band Scatterometry", Report of
 * the European Centre Medium-Range Weather Forecasts (ECMWF), 2003.
 * <p/>
 * [2] C. C. Wackerman, W. G. Pichel, P. Clemente-Colon, "Automated Estimation of Wind Vectors from SAR",
 * 12th Conference on Interactions of the Sea and Atmosphere, 2003.
 */

@OperatorMetadata(alias = "Wind-Field-Estimation", category = "Radar/Feature Extraction/Ocean Applications", authors = "Jun Lu, Luis Veci", version = "1.0", copyright = "Copyright (C) 2015 by Array Systems Computing Inc.", description = "Estimate wind speed and direction")
public class WindFieldEstimationOp extends Operator {

    @SourceProduct(alias = "source")
    private Product sourceProduct;
    @TargetProduct
    private Product targetProduct = null;

    @Parameter(description = "The list of source bands.", alias = "sourceBands", rasterDataNodeType = Band.class, label = "Source Bands")
    private String[] sourceBandNames = null;

    @Parameter(description = "Window size", defaultValue = "20.0", label = "Window Size (km)")
    private double windowSizeInKm = 20.0;

    private String mission = null;
    private int windowSize = 0;
    private int halfWindowSize = 0;
    private int sourceImageWidth = 0;
    private int sourceImageHeight = 0;

    private double rangeSpacing = 0.0;
    private double azimuthSpacing = 0.0;

    private TiePointGrid latitudeTPG = null;
    private TiePointGrid longitudeTPG = null;
    private TiePointGrid incidenceAngle = null;

    private MetadataElement absRoot = null;
    private File windFieldReportFile = null;
    private boolean windFieldEstimated = false;
    private final HashMap<String, List<WindFieldRecord>> bandWindFieldRecord = new HashMap<>();

    @Override
    public void initialize() throws OperatorException {
        try {
            final InputProductValidator validator = new InputProductValidator(sourceProduct);
            validator.checkIfCalibrated(true);
            validator.checkIfTOPSARBurstProduct(false);

            absRoot = AbstractMetadata.getAbstractedMetadata(sourceProduct);

            getMission();

            checkCalibrationFlag();

            getPixelSpacing();

            computeWindowSize();

            getSourceImageDimension();

            getTiePointGrid();

            setTargetReportFilePath();

            createTargetProduct();

        } catch (Throwable e) {
            OperatorUtils.catchOperatorException(getId(), e);
        }
    }

    /**
     * Get mission from the metadata of the product.
     */
    private void getMission() {
        mission = absRoot.getAttributeString(AbstractMetadata.MISSION);
        if (!mission.equals("ERS") && !mission.equals("ENVISAT") && !mission.equals("ERS1")
                && !mission.equals("ERS2") && !mission.contains("SENTINEL-1") && !mission.contains("RS2")) {
            throw new OperatorException("Currently only C-Band SAR products are supported");
        }
    }

    /**
     * Check calibration flag from the metadata of the product.
     *
     * @throws Exception The exceptions.
     */
    private void checkCalibrationFlag() throws Exception {
        if (!AbstractMetadata.getAttributeBoolean(absRoot, AbstractMetadata.abs_calibration_flag)) {
            throw new OperatorException("The product must be calibrated first.");
        }
    }

    /**
     * Get the range and azimuth spacings (in meter).
     *
     * @throws Exception when metadata is missing or equal to default no data value
     */
    private void getPixelSpacing() throws Exception {

        rangeSpacing = AbstractMetadata.getAttributeDouble(absRoot, AbstractMetadata.range_spacing);
        azimuthSpacing = AbstractMetadata.getAttributeDouble(absRoot, AbstractMetadata.azimuth_spacing);
        //System.out.println("Range spacing is " + rangeSpacing);
        //System.out.println("Azimuth spacing is " + azimuthSpacing);
    }

    private void computeWindowSize() {
        windowSize = (int) (windowSizeInKm * 1000 / Math.min(rangeSpacing, azimuthSpacing));
        halfWindowSize = windowSize / 2;
    }

    private void getSourceImageDimension() {
        sourceImageWidth = sourceProduct.getSceneRasterWidth();
        sourceImageHeight = sourceProduct.getSceneRasterHeight();
    }

    /**
     * Get latitude anf longitude tie point grid.
     */
    private void getTiePointGrid() {
        latitudeTPG = OperatorUtils.getLatitude(sourceProduct);
        longitudeTPG = OperatorUtils.getLongitude(sourceProduct);
        incidenceAngle = OperatorUtils.getIncidenceAngle(sourceProduct);
    }

    /**
     * Set absolute path for outputing target report file.
     */
    private void setTargetReportFilePath() {
        final String fileName = sourceProduct.getName() + "_wind_field_report.xml";
        windFieldReportFile = new File(ResourceUtils.getReportFolder(), fileName);
    }

    /**
     * Create target product.
     */
    void createTargetProduct() {

        targetProduct = new Product(sourceProduct.getName(), sourceProduct.getProductType(),
                sourceProduct.getSceneRasterWidth(), sourceProduct.getSceneRasterHeight());

        ProductUtils.copyProductNodes(sourceProduct, targetProduct);

        addSelectedBands();

        updateTargetProductMetadata();
    }

    /**
     * Add the user selected bands to target product.
     *
     * @throws OperatorException The exceptions.
     */
    private void addSelectedBands() throws OperatorException {

        final Band[] sourceBands = OperatorUtils.getSourceBands(sourceProduct, sourceBandNames, false);

        for (Band srcBand : sourceBands) {
            final String srcBandName = srcBand.getName();
            final String unit = srcBand.getUnit();
            if (unit == null) {
                throw new OperatorException("band " + srcBandName + " requires a unit");
            }

            final Band targetBand = new Band(srcBandName, srcBand.getDataType(), sourceImageWidth,
                    sourceImageHeight);

            targetBand.setUnit(unit);
            targetProduct.addBand(targetBand);
            bandWindFieldRecord.put(srcBandName, new ArrayList<>());
        }
    }

    /**
     * Save wind field report file path in the metadata.
     */
    private void updateTargetProductMetadata() {
        final MetadataElement absTgt = AbstractMetadata.getAbstractedMetadata(targetProduct);
        absTgt.setAttributeString(AbstractMetadata.wind_field_report_file, windFieldReportFile.getAbsolutePath());
    }

    /**
     * Called by the framework in order to compute a tile for the given target band.
     * <p>The default implementation throws a runtime exception with the message "not implemented".</p>
     *
     * @param targetBand The target band.
     * @param targetTile The current tile associated with the target band to be computed.
     * @param pm         A progress monitor which should be used to determine computation cancelation requests.
     * @throws OperatorException If an error occurs during computation of the target raster.
     */
    @Override
    public void computeTile(Band targetBand, Tile targetTile, ProgressMonitor pm) throws OperatorException {

        final Rectangle targetTileRectangle = targetTile.getRectangle();
        final int tx0 = targetTileRectangle.x;
        final int ty0 = targetTileRectangle.y;
        final int tw = targetTileRectangle.width;
        final int th = targetTileRectangle.height;
        //System.out.println("tx0 = " + tx0 + ", ty0 = " + ty0 + ", tw = " + tw + ", th = " + th);

        final String targetBandName = targetBand.getName();
        final List<WindFieldRecord> windFieldRecordList = bandWindFieldRecord.get(targetBandName);

        final Band sourceBand = sourceProduct.getBand(targetBandName);
        final double noDataValue = sourceBand.getNoDataValue();
        final String pol = OperatorUtils.getBandPolarization(targetBandName, absRoot);
        Tile sourceTile;

        if (mission.equals("ENVISAT")) {
            if (pol != null && !pol.contains("hh") && !pol.contains("vv")) {
                throw new OperatorException("Polarization " + pol + " is not supported. Please select HH or VV.");
            }
        }
        final Unit.UnitType bandUnit = Unit.getUnitType(sourceBand);
        if (bandUnit != Unit.UnitType.INTENSITY && bandUnit != Unit.UnitType.INTENSITY_DB) {
            throw new OperatorException(
                    "Please select calibrated amplitude or intensity band for wind field estimation");
        }

        // copy the original band data
        targetTile.setRawSamples(getSourceTile(sourceBand, targetTile.getRectangle()).getRawSamples());

        final boolean normlizeSigma = (mission.equals("ENVISAT") && pol.contains("hh"));

        // loop through the center pixel of each frame in the target tile
        int xStart = halfWindowSize;
        while (xStart < tx0) {
            xStart += windowSize;
        }

        int yStart = halfWindowSize;
        while (yStart < ty0) {
            yStart += windowSize;
        }

        final int maxY = ty0 + th;
        final int maxX = tx0 + tw;
        final int halfWindowArea = windowSize * windowSize / 2;
        final int arrowSize = halfWindowSize * 2 / 3;
        for (int y = yStart; y < maxY; y += windowSize) {
            for (int x = xStart; x < maxX; x += windowSize) {

                // get source data for the frame
                final Rectangle sourceTileRectangle = getSourceRectangle(x, y);
                if (sourceTileRectangle == null) {
                    continue;
                }

                final double lat = latitudeTPG.getPixelDouble(x, y);
                final double lon = longitudeTPG.getPixelDouble(x, y);
                final double theta = incidenceAngle.getPixelDouble(x, y);

                sourceTile = getSourceTile(sourceBand, sourceTileRectangle);
                final int numLandPixels = getNumLandPixels(sourceTile, noDataValue);
                if (numLandPixels >= halfWindowArea) {
                    continue;
                }

                final double nrcs = getNormalizedRadarCrossSection(sourceTile, bandUnit, x, y, normlizeSigma,
                        theta);

                // estimate wind direction for the frame
                final double[] direction = { 0.0, 0.0 };
                double ratio = estimateWindDirection(sourceTile, numLandPixels, noDataValue, direction);
                /*
                if (ratio < 0.2 || ratio > 0.8) { 
                continue;
                }
                */
                // estimate wind speed for the frame
                final double speed = estimateWindSpeed(nrcs, direction, theta);

                // save wind field info
                final WindFieldRecord record = new WindFieldRecord(lat, lon, speed, arrowSize * direction[0],
                        arrowSize * direction[1], ratio);

                windFieldRecordList.add(record);
            }
        }

        windFieldEstimated = true;
    }

    /**
     * Get the source tile rectangle centered at a given point.
     *
     * @param x The x coordinate of the given pixel.
     * @param y The y coordinate of the given pixel.
     * @return The rectangle.
     */
    private Rectangle getSourceRectangle(final int x, final int y) {
        final int x0 = x - halfWindowSize;
        final int y0 = y - halfWindowSize;
        final int w = windowSize;
        final int h = windowSize;

        if (x0 < 0 || y0 < 0 || x0 + w > sourceImageWidth || y0 + h > sourceImageHeight) {
            return null;
        }

        return new Rectangle(x0, y0, w, h);
    }

    /**
     * Get the number of land exists in the given window.
     *
     * @param sourceTile  The source tile.
     * @param noDataValue The NoDataValue for the source band.
     * @return The number of land pixels.
     */
    private int getNumLandPixels(final Tile sourceTile, final double noDataValue) {

        final Rectangle sourceTileRectangle = sourceTile.getRectangle();
        final int x0 = sourceTileRectangle.x;
        final int y0 = sourceTileRectangle.y;
        final int w = sourceTileRectangle.width;
        final int h = sourceTileRectangle.height;
        if (w != windowSize || h != windowSize) {
            throw new OperatorException("Source tile size does not match window size.");
        }

        final int maxY = y0 + windowSize;
        final int maxX = x0 + windowSize;
        int numberOfLandPixels = 0;
        for (int y = y0; y < maxY; y++) {
            for (int x = x0; x < maxX; x++) {
                if (sourceTile.getDataBuffer()
                        .getElemDoubleAt(sourceTile.getDataBufferIndex(x, y)) == noDataValue) {
                    numberOfLandPixels++;
                }
            }
        }

        return numberOfLandPixels;
    }

    /**
     * Compute normalized radar cross section for given pixel.
     *
     * @param sourceTile     The source tile.
     * @param bandUnit       The source band unit.
     * @param x              The X coordinate for the given pixel.
     * @param y              The Y coordinate for the given pixel.
     * @param normalizeSigma if mission.contains("ENVISAT") && pol.contains("hh")
     * @param theta          The incidence angle in degree.
     * @return The normalized radar cross section.
     */
    private static double getNormalizedRadarCrossSection(final Tile sourceTile, final Unit.UnitType bandUnit,
            final int x, final int y, final boolean normalizeSigma, final double theta) {

        double sigma = sourceTile.getDataBuffer().getElemDoubleAt(sourceTile.getDataBufferIndex(x, y));
        if (bandUnit == Unit.UnitType.INTENSITY_DB) {
            sigma = FastMath.pow(10.0, sigma / 10);
        }

        if (normalizeSigma) {
            final double alpha = 1.0; // in range [0.4, 1.0]
            final double tanTheta = FastMath.tan(theta * Constants.DTOR);
            sigma *= FastMath.pow((1 + 2.0 * tanTheta * tanTheta) / (1 + alpha * tanTheta * tanTheta), 2.0);
        }
        return sigma;
    }

    /**
     * Estimate wind direction for a given window.
     *
     * @param sourceTile    The source tile.
     * @param numLandPixels The number of land pixels in a given window.
     * @param noDataValue   Source band No Data Value.
     * @param direction     The direction vector.
     * @return ratio The ratio of the minimum quadratic coefficient of the 2D polynomial over the maximum coefficient.
     */
    private double estimateWindDirection(final Tile sourceTile, final int numLandPixels, final double noDataValue,
            final double[] direction) {

        // 1. For each window within which a wind direction will be estimated, a local FFT size is determined.
        //    The FFT size is 2/3 of the window size, therefore four spectra can be computed in the window with
        //    each spectra region has a 50% overlap with the neighboring spectrum.
        //
        // 2. Each window is flattened by applying a large average filter, then dividing by the filtered image.
        //    The filter size for this implementation is set to 11x11.
        //
        // 3. The FFTs are applied and the four resulting spectra are averaged.
        //
        // 4. An annulus is applied to the spectrum to zero out any energy outside of a wavenumber region.
        //    The limits of the annulus are set to wave lengths of 3 km to 15 km.
        //
        // 5. A 3x3 median filter is then applied to the spectrum to remove noise.
        //
        // 6. A 2D polynomial is fit to the resulting spectral samples and the direction through the origin
        //    which has the largest quadratic term (i.e. the widest extent) is determined. The wind direction
        //    is then assumed to be 90 degree from this direction.

        final double[][] imagette = new double[windowSize][windowSize];
        getImagette(sourceTile, numLandPixels, noDataValue, imagette);

        final double[][] dcRemovedImage = new double[windowSize][windowSize];
        removeDCComponent(imagette, dcRemovedImage);

        int fftSize = windowSize * 2 / 3;
        if (fftSize % 2 == 0) {
            fftSize++;
        }

        final double[][] spec = new double[fftSize][fftSize];
        computeSpectrum(dcRemovedImage, fftSize, spec);

        final double delta_k = 1.0 / (windowSizeInKm * 1000.0 * 2.0 / 3.0); // 1 / window_size_in_m
        final int n3 = Math.min((int) (Constants.TWO_PI / (2500.0 * delta_k)), fftSize / 2);
        final RenderedImage annulusAppliedSpec = applyAnnulusToSpec(spec, fftSize, n3, delta_k);

        final RenderedImage filteredImage = medianFilteringSpec(annulusAppliedSpec);

        final double[][] array = new double[2 * n3 + 1][2 * n3 + 1];
        final double peakValue = getPeakSpectrumValue(filteredImage, array, n3);

        return getDirection(array, peakValue, n3, direction);
    }

    private void getImagette(final Tile sourceTile, final int numLandPixels, final double noDataValue,
            final double[][] imagette) {

        final Rectangle sourceTileRectangle = sourceTile.getRectangle();
        final int x0 = sourceTileRectangle.x;
        final int y0 = sourceTileRectangle.y;
        final int w = sourceTileRectangle.width;
        final int h = sourceTileRectangle.height;
        if (w != windowSize || h != windowSize) {
            throw new OperatorException("Source tile size does not match window size.");
        }

        final int maxY = y0 + windowSize;
        final int maxX = x0 + windowSize;
        if (numLandPixels > 0) {

            double mean = 0.0;
            for (int y = y0; y < maxY; y++) {
                for (int x = x0; x < maxX; x++) {
                    final double v = sourceTile.getDataBuffer()
                            .getElemDoubleAt(sourceTile.getDataBufferIndex(x, y));
                    if (v != noDataValue) {
                        mean += v;
                    }
                }
            }
            mean /= windowSize * windowSize - numLandPixels;

            for (int y = y0; y < maxY; y++) {
                for (int x = x0; x < maxX; x++) {
                    final double v = sourceTile.getDataBuffer()
                            .getElemDoubleAt(sourceTile.getDataBufferIndex(x, y));
                    if (v == noDataValue) {
                        imagette[y - y0][x - x0] = mean;
                    } else {
                        imagette[y - y0][x - x0] = v;
                    }
                }
            }

        } else {

            for (int y = y0; y < maxY; y++) {
                for (int x = x0; x < maxX; x++) {
                    imagette[y - y0][x - x0] = sourceTile.getDataBuffer()
                            .getElemDoubleAt(sourceTile.getDataBufferIndex(x, y));
                }
            }
        }

        /*
        // generate simulated data for test
        Random generator = new Random();
        for (int y = y0; y < y0 + windowSize; y++) {
        int r = y - y0;
        for (int x = x0; x < x0 + windowSize; x++) {
            int c = x - x0;
        //                if (r == 30 || r == 60) {
        //                if (c == 30 || c == 60) {
            if (c == r) {
        //                if (c == windowSize - 1 - r) {
                imagette[r][c] = 20*generator.nextDouble();
            } else {
                imagette[r][c] = generator.nextDouble();
            }
        }
        }
        */
        //        dumpData("Imagette", imagette);
    }

    private void removeDCComponent(final double[][] imagette, final double[][] dcRemovedImage) {

        final int filter_size = 11;
        final int half_filter_size = filter_size / 2;
        for (int r = 0; r < windowSize; r++) {
            final int rMin = Math.max(r - half_filter_size, 0);
            final int rMax = Math.min(r + half_filter_size, windowSize - 1);
            for (int c = 0; c < windowSize; c++) {
                final int cMin = Math.max(c - half_filter_size, 0);
                final int cMax = Math.min(c + half_filter_size, windowSize - 1);
                dcRemovedImage[r][c] = imagette[r][c] / getMean(imagette, rMin, rMax, cMin, cMax);
            }
        }
    }

    private static double getMean(final double[][] imagette, final int rMin, final int rMax, final int cMin,
            final int cMax) {

        double mean = 0.0;
        for (int r = rMin; r <= rMax; r++) {
            for (int c = cMin; c <= cMax; c++) {
                mean += imagette[r][c];
            }
        }
        return mean / ((rMax - rMin + 1) * (cMax - cMin + 1));
    }

    private void computeSpectrum(final double[][] srcImage, final int fftSize, final double[][] spec) {

        final double[][] F1 = new double[fftSize][fftSize];
        final double[][] F2 = new double[fftSize][fftSize];
        final double[][] F3 = new double[fftSize][fftSize];
        final double[][] F4 = new double[fftSize][fftSize];
        perform2DFFT(srcImage, 0, fftSize - 1, 0, fftSize - 1, F1);
        perform2DFFT(srcImage, 0, fftSize - 1, windowSize - fftSize, windowSize - 1, F2);
        perform2DFFT(srcImage, windowSize - fftSize, windowSize - 1, 0, fftSize - 1, F3);
        perform2DFFT(srcImage, windowSize - fftSize, windowSize - 1, windowSize - fftSize, windowSize - 1, F4);

        for (int r = 0; r < fftSize; r++) {
            for (int c = 0; c < fftSize; c++) {
                spec[r][c] = (F1[r][c] + F2[r][c] + F3[r][c] + F4[r][c]) / 4.0;
            }
        }
    }

    private static void perform2DFFT(final double[][] srcImage, final int xMin, final int xMax, final int yMin,
            final int yMax, final double[][] spec) {

        // perform 1-D FFT to each row
        final int rowFFTSize = xMax - xMin + 1;
        final int colFFTSize = yMax - yMin + 1;
        final DoubleFFT_1D row_fft = new DoubleFFT_1D(rowFFTSize);
        final double[][] complexDataI = new double[colFFTSize][rowFFTSize];
        final double[][] complexDataQ = new double[colFFTSize][rowFFTSize];
        final double[] rowArray = new double[2 * rowFFTSize];
        for (int y = yMin; y <= yMax; y++) {
            int k = 0;
            for (int x = xMin; x <= xMax; x++) {
                rowArray[k++] = srcImage[y][x];
                rowArray[k++] = 0.0;
            }
            row_fft.complexForward(rowArray);
            for (int c = 0; c < rowFFTSize; c++) {
                complexDataI[y - yMin][c] = rowArray[c + c];
                complexDataQ[y - yMin][c] = rowArray[c + c + 1];
            }
        }
        // dumpData("complexDataI after row FFT", complexDataI);
        // dumpData("complexDataQ after row FFT", complexDataQ);

        // perform 1-D FFT to each column
        final DoubleFFT_1D col_fft = new DoubleFFT_1D(colFFTSize);
        final double[] colArray = new double[2 * colFFTSize];
        for (int x = xMin; x <= xMax; x++) {
            int k = 0;
            for (int y = yMin; y <= yMax; y++) {
                colArray[k++] = complexDataI[y - yMin][x - xMin];
                colArray[k++] = complexDataQ[y - yMin][x - xMin];
            }
            col_fft.complexForward(colArray);
            for (int r = 0; r < colFFTSize; r++) {
                complexDataI[r][x - xMin] = colArray[r + r];
                complexDataQ[r][x - xMin] = colArray[r + r + 1];
            }
        }
        // dumpData("complexDataI after col FFT", complexDataI);
        // dumpData("complexDataQ after col FFT", complexDataQ);

        // get spectrum magnitude and perform fftshift
        final int secondHalfColFFTSize = colFFTSize / 2;
        final int firstHalfColFFTSize = colFFTSize - secondHalfColFFTSize;
        final int secondHalfRowFFTSize = rowFFTSize / 2;
        final int firstHalfRowFFTSize = rowFFTSize - secondHalfRowFFTSize;
        int rr, cc;
        for (int y = yMin; y <= yMax; y++) {
            int r = y - yMin;
            if (r < firstHalfColFFTSize) {
                rr = r + secondHalfColFFTSize;
            } else {
                rr = r - firstHalfColFFTSize;
            }

            for (int x = xMin; x <= xMax; x++) {
                int c = x - xMin;
                if (c < firstHalfRowFFTSize) {
                    cc = c + secondHalfRowFFTSize;
                } else {
                    cc = c - firstHalfRowFFTSize;
                }
                spec[rr][cc] = complexDataI[r][c] * complexDataI[r][c] + complexDataQ[r][c] * complexDataQ[r][c];
            }
        }
        // dumpData("spec", spec);
    }

    private static RenderedImage createRenderedImage(double[] array, int width, int height) {

        // create rendered image with demension being width by height
        final SampleModel sampleModel = RasterFactory.createBandedSampleModel(DataBuffer.TYPE_DOUBLE, width, height,
                1);
        final ColorModel colourModel = PlanarImage.createColorModel(sampleModel);
        final DataBufferDouble dataBuffer = new DataBufferDouble(array, array.length);
        final WritableRaster raster = RasterFactory.createWritableRaster(sampleModel, dataBuffer, new Point(0, 0));
        return new BufferedImage(colourModel, raster, false, new Hashtable());
    }

    private static RenderedImage applyAnnulusToSpec(final double[][] spec, final int fftSize, final int n3,
            final double delta_k) {

        final int halfFFTSize = fftSize / 2;
        int n15 = (int) (Constants.TWO_PI / (15000.0 * delta_k));
        if (n15 >= halfFFTSize) {
            n15 = 1;
        }
        final double[] array = new double[(2 * n3 + 1) * (2 * n3 + 1)];
        int k = 0;
        for (int r = halfFFTSize - n3; r < halfFFTSize + n3 + 1; r++) {
            for (int c = halfFFTSize - n3; c < halfFFTSize + n3 + 1; c++) {
                if (r >= halfFFTSize - n15 && r <= halfFFTSize + n15 && c >= halfFFTSize - n15
                        && c <= halfFFTSize + n15) {
                    array[k++] = 0.0;
                } else {
                    array[k++] = spec[r][c];
                }
            }
        }
        //        dumpData("before median filtering:", spec);

        return createRenderedImage(array, 2 * n3 + 1, 2 * n3 + 1);
    }

    private static RenderedImage medianFilteringSpec(final RenderedImage annulusAppliedSpec) {

        final int size = 3;
        final MedianFilterShape shape = MedianFilterDescriptor.MEDIAN_MASK_SQUARE;
        final ParameterBlock pb = new ParameterBlock();
        pb.addSource(annulusAppliedSpec);
        pb.add(shape);
        pb.add(size);
        return JAI.create("medianfilter", pb);
    }

    private static double getPeakSpectrumValue(final RenderedImage filteredImage, final double[][] array,
            final int n3) {

        final Raster data = filteredImage.getData();
        double peakValue = 0.0;
        final int length = 2 * n3 + 1;
        for (int y = 0; y < length; y++) {
            for (int x = 0; x < length; x++) {
                array[y][x] = data.getSampleDouble(x, y, 0);
                if (peakValue < array[y][x]) {
                    peakValue = array[y][x];
                }
            }
        }
        return peakValue;
    }

    /**
     * Compute wind direction by performing 2D polynomial fitting to the spectral samples.
     *
     * @param array     Array holding the spectrum samples.
     * @param peakValue The peak spectrum sample.
     * @param n3        Spectrum size is 2*n3+1.
     * @param direction Wind direction (dx, dy).
     * @return The ratio of the minor semi axes over the major semi axes of the 2D polynomial.
     */
    private static double getDirection(final double[][] array, final double peakValue, final int n3,
            final double[] direction) {

        //        dumpData("spec", array);

        double m00 = 0.0, m01 = 0.0, m02 = 0.0;
        double m10 = 0.0, m11 = 0.0, m12 = 0.0;
        double m20 = 0.0, m21 = 0.0, m22 = 0.0;
        double s0 = 0.0, s1 = 0.0, s2 = 0.0;

        final int length = 2 * n3 + 1;
        for (int y = 0; y < length; y++) {
            final int yy = y - n3;
            for (int x = 0; x < length; x++) {
                final int xx = x - n3;
                final double v = array[y][x] - peakValue;
                m00 += xx * xx * xx * xx;
                m01 += xx * xx * yy * yy;
                m02 += xx * xx * xx * yy;
                m11 += yy * yy * yy * yy;
                m12 += xx * yy * yy * yy;
                s0 += xx * xx * v;
                s1 += yy * yy * v;
                s2 += xx * yy * v;
            }
        }

        m10 = m01;
        m20 = m02;
        m21 = m12;
        m22 = m01;

        final Matrix M = new Matrix(3, 3);
        M.set(0, 0, m00);
        M.set(0, 1, m01);
        M.set(0, 2, m02);
        M.set(1, 0, m10);
        M.set(1, 1, m11);
        M.set(1, 2, m12);
        M.set(2, 0, m20);
        M.set(2, 1, m21);
        M.set(2, 2, m22);

        final Matrix s = new Matrix(3, 1);
        s.set(0, 0, s0);
        s.set(1, 0, s1);
        s.set(2, 0, s2);

        final Matrix c = M.solve(s);
        final double c0 = c.get(0, 0); // c0*x^2 + c1*y^2 + c2*x*y
        final double c1 = c.get(1, 0);
        final double c2 = -c.get(2, 0); // flip y axis pointing up

        double d = Math.sqrt((c0 - c1) * (c0 - c1) + c2 * c2);
        double d2 = 2.0 * d;
        double tmp = Math.abs(c0 - c1);
        double cos_theta_2 = (d + tmp) / d2;
        double sin_theta_2 = (d - tmp) / d2;
        double sin_cos = c2 * tmp / ((c0 - c1) * d2);
        double a = (c0 * (d + tmp) + c1 * (d - tmp) + c2 * c2 * tmp / (c0 - c1)) / d2;
        double b = (c0 * (d - tmp) + c1 * (d + tmp) - c2 * c2 * tmp / (c0 - c1)) / d2;

        if (cos_theta_2 == 0.0) {
            if (Math.abs(a) > Math.abs(b)) {
                direction[0] = 1.0;
                direction[1] = 0.0;
            } else {
                direction[0] = 0.0;
                direction[1] = 1.0;
            }
        } else if (sin_theta_2 == 0.0) {
            if (Math.abs(a) > Math.abs(b)) {
                direction[0] = 0.0;
                direction[1] = 1.0;
            } else {
                direction[0] = 1.0;
                direction[1] = 0.0;
            }
        } else {
            double k = (sin_cos / Math.abs(sin_cos)) * Math.sqrt(sin_theta_2 / cos_theta_2);
            if (k > 0) {
                if (Math.abs(a) > Math.abs(b)) {
                    direction[0] = -k / Math.sqrt(1 + k * k);
                    direction[1] = 1 / Math.sqrt(1 + k * k);
                } else {
                    direction[0] = 1 / Math.sqrt(1 + k * k);
                    direction[1] = k / Math.sqrt(1 + k * k);
                }
            } else { // k < 0
                if (Math.abs(a) > Math.abs(b)) {
                    direction[0] = -k / Math.sqrt(1 + k * k);
                    direction[1] = 1 / Math.sqrt(1 + k * k);
                } else {
                    direction[0] = -1 / Math.sqrt(1 + k * k);
                    direction[1] = -k / Math.sqrt(1 + k * k);
                }
            }
        }

        // wind direction is 90 degree from this direction
        tmp = direction[0];
        direction[0] = -direction[1];
        direction[1] = tmp;

        return Math.min(Math.abs(a), Math.abs(b)) / Math.max(Math.abs(a), Math.abs(b));
    }

    /**
     * Dump data. This function is for debugging use only.
     *
     * @param title The title of the data.
     * @param data  2-D array holding the data.
     */
    private synchronized static void dumpData(final String title, final double[][] data) {
        System.out.println();
        System.out.println(title + ";");
        final int h = data.length;
        final int w = data[0].length;
        for (double[] aData : data) {
            for (int c = 0; c < w; c++) {
                System.out.print(aData[c] + ",");
            }
            System.out.println();
        }
        System.out.println();
    }

    /**
     * Estimate wind speed using CMOD5 model.
     *
     * @param nrcs      The normalized radar cross section.
     * @param direction The wind direction vector.
     * @param theta     The incidence angle in degree.
     * @return The wind speed in m/s.
     */
    private static double estimateWindSpeed(final double nrcs, final double[] direction, final double theta) {

        final double fi = Math.atan2(direction[1], direction[0]) * Constants.RTOD;
        final double cosFI = FastMath.cos(fi * Constants.DTOR);

        // try wind speed from 0.1 m/s to 20 m/s with step size 0.1
        final double[] err = new double[200];
        err[0] = Math.abs(nrcs - CMOD5.compute(0.1, cosFI, theta));
        double errMin = err[0];
        int errMinIndex = 0;
        for (int i = 1; i < 200; i++) {
            final double v = (i + 1) * 0.1; // speed
            err[i] = Math.abs(nrcs - CMOD5.compute(v, cosFI, theta));
            if (err[i] < errMin) {
                errMin = err[i];
                errMinIndex = i;
            }
        }

        return (errMinIndex + 1) * 0.1;
    }

    private static class CMOD5 {

        private final static double c1 = -0.688;
        private final static double c15 = 0.007;
        private final static double c2 = -0.793;
        private final static double c16 = 0.33;
        private final static double c3 = 0.338;
        private final static double c17 = 0.012;
        private final static double c4 = -0.173;
        private final static double c18 = 22.0;
        private final static double c5 = 0.0;
        private final static double c19 = 1.95;
        private final static double c6 = 0.004;
        private final static double c20 = 3.0;
        private final static double c7 = 0.111;
        private final static double c21 = 8.39;
        private final static double c8 = 0.0162;
        private final static double c22 = -3.44;
        private final static double c9 = 6.34;
        private final static double c23 = 1.36;
        private final static double c10 = 2.57;
        private final static double c24 = 5.35;
        private final static double c11 = -2.18;
        private final static double c25 = 1.99;
        private final static double c12 = 0.4;
        private final static double c26 = 0.29;
        private final static double c13 = -0.6;
        private final static double c27 = 3.80;
        private final static double c14 = 0.045;
        private final static double c28 = 1.53;

        private final static double THETM = 40.0;
        private final static double THETHR = 25.0;
        private final static double ZPOW = 1.6;

        private final static double y0 = c19;
        private final static double n = c20;
        private final static double a = y0 - (y0 - 1) / n;
        private final static double b = 1 / (n * FastMath.pow(y0 - 1, n - 1));

        /**
         * Compute normalized radar cross section (NRCS) using CMOD5 model.
         *
         * @param v              The wind speed in m/s.
         * @param cosFI          The cos of the angle between radar look direction and wind direction (in degree).
         * @param incidenceAngle The incidence angle in degree.
         * @return The NRCS.
         */
        static double compute(final double v, final double cosFI, final double incidenceAngle) {

            final double x = (incidenceAngle - THETM) / THETHR;
            final double xx = x * x;
            final double a0 = c1 + c2 * x + c3 * xx + c4 * x * xx;
            final double a1 = c5 + c6 * x;
            final double a2 = c7 + c8 * x;
            final double gamma = c9 + c10 * x + c11 * xx;
            final double s0 = c12 + c13 * x;
            final double s = a2 * v;
            double a3 = 1.0 / (1.0 + FastMath.exp(-Math.max(s, s0)));
            if (s < s0) {
                a3 = a3 * FastMath.pow((s / s0), s0 * (1.0 - a3));
            }

            final double b0 = FastMath.pow(a3, gamma) * FastMath.pow(10.0, a0 + a1 * v);
            double b1 = c15 * v * (0.5 + x - FastMath.tanh(4.0 * (x + c16 + c17 * v)));
            b1 = (c14 * (1.0 + x) - b1) / (FastMath.exp(0.34 * (v - c18)) + 1);
            final double v0 = c21 + c22 * x + c23 * xx;
            final double d1 = c24 + c25 * x + c26 * xx;
            final double d2 = c27 + c28 * x;
            double v2 = v / v0 + 1.0;
            if (v2 < y0) {
                v2 = a + b * FastMath.pow(v2 - 1.0, n);
            }
            final double b2 = (-d1 + d2 * v2) * FastMath.exp(-v2);

            return b0 * FastMath.pow(1.0 + b1 * cosFI + b2 * (2.0 * cosFI * cosFI - 1.0), ZPOW);
        }
    }

    /**
     * Output cluster information to file.
     */
    @Override
    public void dispose() {

        if (!windFieldEstimated) {
            return;
        }

        outputWindFieldInfoToFile();
    }

    /**
     * Output wind fielld information to file.
     *
     * @throws OperatorException when can't save metadata
     */
    private void outputWindFieldInfoToFile() throws OperatorException {
        /*
        double dxMean = 0.0;
        double dyMean = 0.0;
        int counter = 0;
        for (String bandName : bandWindFieldRecord.keySet())  {
        final java.util.List<WindFieldRecord> recordList = bandWindFieldRecord.get(bandName);
        for (WindFieldRecord rec : recordList) {
            dxMean += rec.dx;
            dyMean += rec.dy;
            counter++;
        }
        }
        dxMean /= counter;
        dyMean /= counter;
        */
        final Element root = new Element("Detection");
        final Document doc = new Document(root);

        for (String bandName : bandWindFieldRecord.keySet()) {
            final Element elem = new Element("windFieldEstimated");
            elem.setAttribute("bandName", bandName);
            final java.util.List<WindFieldRecord> recordList = bandWindFieldRecord.get(bandName);
            for (WindFieldRecord rec : recordList) {
                /*
                if (rec.dx*dxMean + rec.dy*dyMean <= 0.707) {
                continue;
                }
                */
                final Element subElem = new Element("windFieldInfo");
                subElem.setAttribute("lat", String.valueOf(rec.lat));
                subElem.setAttribute("lon", String.valueOf(rec.lon));
                subElem.setAttribute("speed", String.valueOf(rec.speed));
                subElem.setAttribute("dx", String.valueOf(rec.dx));
                subElem.setAttribute("dy", String.valueOf(rec.dy));
                subElem.setAttribute("ratio", String.valueOf(rec.ratio));
                elem.addContent(subElem);
            }
            root.addContent(elem);
        }
        XMLSupport.SaveXML(doc, windFieldReportFile.getAbsolutePath());
    }

    public static class WindFieldRecord {
        public final double lat;
        public final double lon;
        public final double speed;
        public final double dx;
        public final double dy;
        public final double ratio;

        public WindFieldRecord(final double lat, final double lon, final double speed, final double dx,
                final double dy, final double ratio) {
            this.lat = Math.round(lat * 100.0) / 100.0;
            this.lon = Math.round(lon * 100.0) / 100.0;
            this.speed = Math.round(speed * 100.0) / 100.0;
            this.dx = Math.round(dx * 100.0) / 100.0;
            this.dy = Math.round(dy * 100.0) / 100.0;
            this.ratio = Math.round(ratio * 100.0) / 100.0;
        }
    }

    /**
     * Operator SPI.
     */
    public static class Spi extends OperatorSpi {

        public Spi() {
            super(WindFieldEstimationOp.class);
        }
    }
}