Subversion Repositories mkgmap

Rev

Rev 4653 | Blame | Compare with Previous | Last modification | View Log | RSS feed

/*
 * Copyright (C) 2017.
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 3 or
 * version 2 as published by the Free Software Foundation.
 *
 * 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.
 */

package uk.me.parabola.mkgmap.reader.hgt;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.nio.ByteBuffer;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
import java.util.zip.ZipEntry;
import java.util.zip.ZipFile;

import uk.me.parabola.imgfmt.MapFailedException;
import uk.me.parabola.imgfmt.Utils;
import uk.me.parabola.log.Logger;

import static java.nio.channels.FileChannel.MapMode.READ_ONLY;

/**
 * Rather simple code to read a single HGT file with SRTM data. Based on old code in class HGTDEM
 * in package uk.me.parabola.mkgmap.reader.dem which was removed in 2017.
 * @author Gerd Petermann
 *
 */

public class HGTReader {
        private static final Logger log = Logger.getLogger(HGTReader.class);

        public static final short UNDEF = Short.MIN_VALUE;
        private ByteBuffer buffer;
        private int res;
        private final String fileName;
        private String path;
        private boolean read;
        private long count;
        private int numPixelsX;
        private int numPixelsY;

       
        private static final Map<String,Set<String>> missingMap = new HashMap<>();
        private static final Set<String> badDir = new HashSet<>();
       
        /**
         * Class to read a single HGT file.
         * @param lat in degrees, -90 .. 90
         * @param lon - -180..180
         * @param dirsWithHGT string with comma separated list of directories to search for *.hgt files
         * Supported are also zip files containing *.hgt files and directories containing *.hgt.zip.
         */

        public HGTReader(int lat, int lon, String dirsWithHGT) {
                String baseName = String.format("%s%02d%s%03d",
                                lat < 0 ? "S" : "N", lat < 0 ? -lat : lat,
                                                lon < 0 ? "W" : "E", lon < 0 ? -lon : lon);
               
                String[] dirs = dirsWithHGT.split("[,]");
                fileName = baseName + ".hgt";
                String fName = ".";
                boolean knownAsMissing = false;
                synchronized (missingMap) {
                        Set<String> missingSet = missingMap.get(dirsWithHGT);
                        if (missingSet != null)
                                knownAsMissing = missingSet.contains(fileName);
                }
                if (!knownAsMissing) {
                        for (String dir : dirs) {
                                dir = dir.trim();
                                File f = new File (dir);
                                if (!f.exists()) {
                                        synchronized (badDir) {
                                                if (badDir.add(dir))
                                                        log.error("extracted path >" + dir + "< does not exist, check option dem:", dirsWithHGT);
                                        }
                                        continue;
                                }
                                if (f.isDirectory()) {
                                        fName = Utils.joinPath(dir, fileName);
                                        try (FileInputStream fis = new FileInputStream(fName)) {
                                                res = calcRes(fis.getChannel().size(), fName);
                                                if (res >= 0) {
                                                        path = fName;
                                                }
                                                break;
                                        } catch (FileNotFoundException e) {
                                        } catch (IOException e) {
                                                log.error("failed to get size for file", fName);
                                        }
                                        fName += ".zip";
                                        checkZip(fName, fileName); // try to find *.hgt.zip in dir that contains *.hgt
                                        if (res > 0) {
                                                path = fName;
                                                return;
                                        }
                                        fName = Utils.joinPath(dir, baseName) + ".zip";
                                        checkZip(fName, fileName); // try to find *.hgt.zip in dir that contains *.hgt
                                        if (res > 0) {
                                                path = fName;
                                                return;
                                        }
                                } else {
                                        if (dir.endsWith(".zip")) {
                                                checkZip(dir, fileName); // try to find *.hgt in zip file
                                        }
                                }
                                if (res > 0) {
                                        path = dir;
                                        return;
                                }
                        }
                        if (res <= 0 || path == null) {
                                res = -1;
                                path = null;
                                synchronized (missingMap){
                                        missingMap.computeIfAbsent(dirsWithHGT, k-> new HashSet<>()).add(fileName);
                                }
                                HGTList hgtList = HGTList.get();
                                if (hgtList != null) {
                                        if (hgtList.shouldExist(lat, lon))
                                                Logger.defaultLogger.warn(this.getClass().getSimpleName() + ": file " + fileName + " not found but it should exist. Height values will be 0.");
                                } else {
                                        log.warn("file " + fileName + " not found. Is expected to cover sea.");
                                }
                        }
                }
        }
       
        /**
         * try to find the needed file. Different hgt providers use slightly different methods to
         * pack their data.
         * @param zipFile
         * @param name
         * @return
         */

        private ZipEntry findZipEntry (ZipFile zipFile, String name) {
                ZipEntry entry = zipFile.getEntry(name);
                if (entry == null) {
                        // no direct hit, try to recurse through all files
                        Enumeration<? extends ZipEntry> entries = zipFile.entries();
                        while (entries.hasMoreElements()) {
                                entry = entries.nextElement();
                                if (!entry.isDirectory() && entry.getName().toUpperCase().endsWith(name.toUpperCase())) {
                                        return entry;
                                }
                        }
                        return null;
                }
                return entry;
        }
       
        /**
         * Check if we can find the wanted file in a zip container and if it has the right size.
         * @param fName path to container
         * @param name wanted file
         */

        private void checkZip(String fName, String name) {
                File f = new File(fName);
                if (!f.exists())
                        return;
                try(ZipFile zipFile = new ZipFile(f)){
                        ZipEntry entry = findZipEntry(zipFile, name);
                        if (entry != null){
                                res = calcRes(entry.getSize(), entry.getName());
                        }
                } catch (IOException exp) {
                        log.error("failed to get size for file", name, "from", fName);
                }
        }

        /**
         * Try to unzip the file contained in a zip file.
         * @param zipFile
         * @param entry
         * @throws IOException
         */

        private void extractFromZip(String fName, String name) throws IOException {
                try (ZipFile zipFile = new ZipFile(fName)) {
                        ZipEntry entry = findZipEntry(zipFile, name);
                        if (entry != null) {
                                InputStream is = zipFile.getInputStream(entry);
                                log.info("extracting data for " + entry.getName() + " from " + zipFile.getName());
                                buffer = ByteBuffer.allocate((int) entry.getSize());
                                byte[] ioBuffer = new byte[1024];
                                int len = is.read(ioBuffer);
                                while (len != -1) {
                                        buffer.put(ioBuffer, 0, len);
                                        len = is.read(ioBuffer);
                                }
                        } else {
                                throw new FileNotFoundException(name);
                        }
                        read = true;
                }
        }

        /**
         * calculate the resolution of the hgt file. size should be exactly 2 * (res+1) * (res+1)
         * @param size number of bytes
         * @param fname file name (for error possible message)
         * @return resolution (typically 1200 for 3'' or 3600 for 1'')
         */

        private int calcRes(long size, String fname) {
                long numVals = (long) Math.sqrt(size / 2d);
                if (2 * 1801 * 3601 == size) {
                        numPixelsX = 1801;
                        numPixelsY = 3601;
                        return 3600;
                }
                if (2 * numVals * numVals == size) {
                        numPixelsX = (int) numVals;
                        numPixelsY = (int) numVals;
                        return (int) (numVals - 1);
                }
                log.error("file", fname, "has unexpected size", size, "and is ignored");
                return -1;
        }

        /**
         * HGT files are organised as a matrix of n*n (short) values giving the elevation in metres.
         * Invalid values are coded as 0x8000 = -327678 = Short.MIN_VALUE.
         * @param x index for column west to east
         * @param y index for row north to south
         * @return the elevation value stored in the file or 0 if
         */

        public short ele(int x, int y) {
                if (!read && path != null) {
                        prepRead();
                }
                if (buffer == null)
                        return 0;
                assert (x >= 0 && x < numPixelsX && y >= 0 && y < numPixelsY) :
                        "wrong x/y value for res" + numPixelsX + "x" + numPixelsY + " x=" + x + " y=" + y;
                count++;
                return buffer.getShort(2 * ((numPixelsY - 1 - y) * numPixelsX + x));
               
        }

        /**
         * @return the resolution to use with this file, -1 is return if file is invalid
         */

        public int getRes() {
                return res;
        }
       
        @Override
        public String toString() {
                return fileName + " (" + count + " reads) " + res ;
        }

        /**
         * Return memory to GC.
         * @return true if heap memory was freed.
         */

        public boolean freeBuf() {
                if (buffer == null)
                        return false;
                buffer = null;
                read = false;
                return true;
        }

        public void prepRead() {
                if (!read && path != null) {
                        try {
                                if (count == 0)
                                        log.info("allocating buffer for", fileName);
                                else
                                        log.warn("re-allocating buffer for", fileName);
                                if (path.endsWith(".zip"))
                                        extractFromZip(path, fileName);
                                else {
                                        try (FileInputStream is = new FileInputStream(path)) {
                                                buffer = is.getChannel().map(READ_ONLY, 0, is.getChannel().size());
                                                read = true;
                                        }
                                }
                        } catch (FileNotFoundException e) {
                                throw new MapFailedException("previously existing file is missing: " + path);
                        } catch (IOException e) {
                                log.error("failed to create buffer for file", path);
                        }

                }
        }

        public int getResX() {
                return numPixelsX - 1;
        }

        public int getResY() {
                return numPixelsY - 1;
        }
       
}