From efb99c36faebf82a74e6d4491a810edccd1bad00 Mon Sep 17 00:00:00 2001 From: Nicolas Chiaruttini Date: Fri, 8 Sep 2023 15:14:58 +0200 Subject: [PATCH 01/16] Commits an alternative Zeiss CZI Reader --- .../src/loci/formats/in/ZeissCZIReader.java | 7350 +++++++++-------- .../src/loci/formats/in/libczi/LibCZI.java | 1113 +++ 2 files changed, 4801 insertions(+), 3662 deletions(-) create mode 100644 components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java diff --git a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java index fa704467ee9..4f7088f8518 100644 --- a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java +++ b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java @@ -22,20 +22,32 @@ * . * #L% */ - package loci.formats.in; -import java.io.ByteArrayInputStream; -import java.io.File; -import java.io.IOException; -import java.util.ArrayDeque; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.Deque; -import java.util.HashMap; -import java.util.HashSet; -import java.util.Iterator; -import javax.xml.parsers.DocumentBuilder; +/* + * #%L + * OME Bio-Formats package for reading and converting biological file formats. + * %% + * Copyright (C) 2005 - 2017 Open Microscopy Environment: + * - Board of Regents of the University of Wisconsin-Madison + * - Glencoe Software, Inc. + * - University of Dundee + * %% + * 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 2 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 + * . + * #L% + */ import loci.common.ByteArrayHandle; import loci.common.Constants; @@ -44,6 +56,9 @@ import loci.common.Location; import loci.common.RandomAccessInputStream; import loci.common.Region; +import loci.common.services.DependencyException; +import loci.common.services.ServiceException; +import loci.common.services.ServiceFactory; import loci.common.xml.XMLTools; import loci.formats.CoreMetadata; import loci.formats.FormatException; @@ -55,8 +70,20 @@ import loci.formats.codec.JPEGXRCodec; import loci.formats.codec.LZWCodec; import loci.formats.codec.ZstdCodec; +import loci.formats.in.DynamicMetadataOptions; +import loci.formats.in.JPEGReader; +import loci.formats.in.MetadataOptions; +import loci.formats.in.libczi.LibCZI; import loci.formats.meta.MetadataStore; - +import loci.formats.ome.OMEXMLMetadata; +import loci.formats.services.OMEXMLService; +import ome.units.UNITS; +import ome.units.quantity.Length; +import ome.units.quantity.Power; +import ome.units.quantity.Pressure; +import ome.units.quantity.Temperature; +import ome.units.quantity.Time; +import ome.units.unit.Unit; import ome.xml.model.enums.AcquisitionMode; import ome.xml.model.enums.Binning; import ome.xml.model.enums.IlluminationType; @@ -66,569 +93,828 @@ import ome.xml.model.primitives.PositiveFloat; import ome.xml.model.primitives.PositiveInteger; import ome.xml.model.primitives.Timestamp; - -import ome.units.quantity.Length; -import ome.units.quantity.Power; -import ome.units.quantity.Pressure; -import ome.units.quantity.Temperature; -import ome.units.quantity.Time; -import ome.units.UNITS; - -import org.xml.sax.SAXException; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; import org.w3c.dom.Element; import org.w3c.dom.NamedNodeMap; import org.w3c.dom.Node; import org.w3c.dom.NodeList; +import org.xml.sax.SAXException; + +import javax.xml.parsers.DocumentBuilder; +import java.io.ByteArrayInputStream; +import java.io.File; +import java.io.IOException; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.ref.SoftReference; +import java.lang.reflect.Field; +import java.util.ArrayDeque; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Collection; +import java.util.Comparator; +import java.util.Deque; +import java.util.HashMap; +import java.util.HashSet; +import java.util.Iterator; +import java.util.LinkedHashMap; +import java.util.List; +import java.util.Map; +import java.util.Objects; +import java.util.Set; +import java.util.concurrent.atomic.AtomicLong; +import java.util.concurrent.locks.Lock; +import java.util.concurrent.locks.ReentrantLock; +import java.util.function.Function; +import java.util.stream.Collectors; + +import static loci.formats.in.libczi.LibCZI.JPEG; +import static loci.formats.in.libczi.LibCZI.JPEGXR; +import static loci.formats.in.libczi.LibCZI.LZW; +import static loci.formats.in.libczi.LibCZI.UNCOMPRESSED; +import static loci.formats.in.libczi.LibCZI.ZSTD_0; +import static loci.formats.in.libczi.LibCZI.ZSTD_1; /** * ZeissCZIReader is the file format reader for Zeiss .czi files. + * See @see CZI reference documentation + *

+ * Essentially, all data is stored into subblocks where each subblock location is specified by its dimension indices. + * There are standard spatial and time dimensions, as well as extra ones necessary to describe channel, scenes, + * acquisition modalities, etc: + *

+ * X,Y,Z, // 3 spaces dimension + * T, Time + * M, Mosaic but why is there no trace of it in libczi documentation ??? + * C, Channel + * R, Rotation + * I, Illumination + * H, Phase + * V, View + * B, Block = deprecated + * S Scene + *

+ * + * A subblock may represent a lower resolution level. How to know this ? Because its stored size (x or y) is lower + * than its size (x or y). Its downscaling factor can thus be computed the ratio between stored size and size. + * For convenience, this reader adds the downscaling factor as an extra dimension named 'PY' + *

+ * A CZI file consists of several segments. The majority of segments are data subblocks, as described before. But other + * segments are present. Essentially this reader reads the {@link LibCZI.FileHeaderSegment} that + * contains some metadata as well as the location of the {@link LibCZI.SubBlockDirectorySegment} + * + * The SubBlockDirectorySegment is a critical segment because it contains the dimension indices and file location of all + * data subblocks. Thus, by reading this segment only, there is no need to go through all file segments while + * initializing the reader. + *

+ * Using this initial reading of the directory segment, all dimensions and all dimension ranges are known in advance. + * This is used to compute the number of core series of the reader, as well as the resolution levels. This is done + * by creating a core series signature {@link CoreSignature} where the dimension are sorted according to a priority + * {@link ZeissCZIReader#dimensionPriority(String)}. If autostitching is true, all mosaics belong to the same core + * series. If autostitching is false, each mosaic is split into different core series. + *

+ * (core series (or core index) = series + resolution level) + *

+ * Notes: + * 1. It is assumed that all subblocks from a single core index + * have the same compression type {@link ZeissCZIReader#coreIndexToCompression} + * + * 2. This reader is not thread safe, you can use memoization or {@link ZeissCZIReader#copy()} + * to get a new reader and perform parallel reading. + *

+ * 3. This reader is optimized for low memory footprint. It has been tested to work on Tb + * czi size files. To save memory, the data structures used for reading are trimmed to the minimal amount of data + * necessary for the reading after the reader has been initialized To illustrate this point, for a 6Tb dataset, each 'int' + * saved per block saves 7Mb (in RAM and in memo file). Trimming down libczi dimension entries + * to {@link MinDimEntry} leads to a memo file of around 100Mb for a 4Tb czi file. Its initialisation + * takes below a minute, with memo building. Then a few seconds to generate a new reader from a memo file is sufficient. + *

+ * 4. Even with memoization, at runtime, a reader for a multi Tb file will take around 300Mb on the heap. While this + * is reasonable for a single reader, it becomes an issue to create multiple readers for parallel reading: 10 readers + * will take 3 Gb. Thus the method {@link ZeissCZIReader#copy()} exist in order to create a new reader from an + * existing one, which saves memory because it reuses all fields from the previous reader. Using this method, 10 + * readers can be created to read in parallel en single czi file, but it will use only the memory of one reader. + * WARNING: calling {@link ZeissCZIReader#close()} on one of these readers will prevent the use + * of all the other readers created with the copy method! + *

+ * 5. This reader has an optimisation for lattice-light sheet fast reader initialisation: the metadata for each subblock + * is not read for each plane, but only for the first plane of each time point, and then the timestamp of each plane is + * linearly interpolated. This prevents a lot of random accesses to the file when it is initialized (up to a factor 1000 + * if there are 1000 planes). This a small cost associated to this choice: the linear interpolation is not alway exact, + * and timestamps shift up to 10 ms may exist. + *

+ * The annotation {@link CopyByRef} is used to annotate the fields that should be initialized in the duplicated reader + * using the reference of the model one, see the constructor with the reader in argument. + *

+ * 5. This reader uses the class {@link LibCZI} which contains the czi data structure translated to Java and which + * contains very no logic related to the reader itself. + *

+ * TODO: + * - get optimal tile size should vary depending on compression: on raw data it's easy to partially read planes, + * but for compressed data that's much harder so it would be better to read the whole block rather that decompressing + * it multiple times the same block to extract a partial region. + * Missing features: + * - add two methods that map forth and back czi dimension indices to bio-formats series + * - add a method that returns a 3D matrix per series (for lattice skewed dataset?) take care with version + * Issues: + * - some absolute path are stored in the reader, thus the memo fails if the file is moved + * - improve: slide preview and label image are stored directly in the reader as a byte array. That does not look optimal + * but loading these bytes on demand is quite tedious: hard to explain, but a reader is created inside the reader and + * maps the file 'temporarily' to a fake file. That's pretty clever and convenient, but prevents (most probably) + * lazy loading AND memoization functionality. + * + * TODO: sync https://github.com/ome/bioformats/pull/4088, that was fixed after this reader was branched from bio-formats + * TODO: implement getfillcolor + * TODO: test PALM file + * TODO: ask how to get rid of absolute file path in memo that do not crash the reader when the file is moved + * */ + public class ZeissCZIReader extends FormatReader { - // -- Constants -- + final static Logger logger = LoggerFactory.getLogger(ZeissCZIReader.class); - public static final String ALLOW_AUTOSTITCHING_KEY = - "zeissczi.autostitch"; + // -- Constants -- + public static final String ALLOW_AUTOSTITCHING_KEY = "zeissczi.autostitch"; public static final boolean ALLOW_AUTOSTITCHING_DEFAULT = true; - public static final String INCLUDE_ATTACHMENTS_KEY = - "zeissczi.attachments"; + public static final String INCLUDE_ATTACHMENTS_KEY = "zeissczi.attachments"; public static final boolean INCLUDE_ATTACHMENTS_DEFAULT = true; public static final String TRIM_DIMENSIONS_KEY = "zeissczi.trim_dimensions"; public static final boolean TRIM_DIMENSIONS_DEFAULT = false; public static final String RELATIVE_POSITIONS_KEY = "zeissczi.relative_positions"; public static final boolean RELATIVE_POSITIONS_DEFAULT = false; - - private static final int ALIGNMENT = 32; - private static final int HEADER_SIZE = 32; private static final String CZI_MAGIC_STRING = "ZISRAWFILE"; private static final int BUFFER_SIZE = 512; - /** Compression constants. */ - private static final int UNCOMPRESSED = 0; - private static final int JPEG = 1; - private static final int LZW = 2; - private static final int JPEGXR = 4; - private static final int ZSTD_0 = 5; - private static final int ZSTD_1 = 6; - - /** Pixel type constants. */ - private static final int GRAY8 = 0; - private static final int GRAY16 = 1; - private static final int GRAY_FLOAT = 2; - private static final int BGR_24 = 3; - private static final int BGR_48 = 4; - private static final int BGR_FLOAT = 8; - private static final int BGRA_8 = 9; - private static final int COMPLEX = 10; - private static final int COMPLEX_FLOAT = 11; - private static final int GRAY32 = 12; - private static final int GRAY_DOUBLE = 13; + // A string identifier for an extra dimension: the resolution level. It's not directly part of the CZI format, + // at least not written as a dimension entry + private static final String RESOLUTION_LEVEL_DIMENSION = "PY"; + + // A string identifier for an extra dimension: the file part. It's not directly part of the CZI format, + // at least not written as a dimension entry + private static final String FILE_PART_DIMENSION = "PA"; // -- Fields -- + // bio-formats core index to x origin, in the Zeiss 2D coordinates system, common to all planes. Unit: pixel (highest resolution level) + @CopyByRef + private List coreIndexToOx = new ArrayList<>(); + + // bio-formats core index to y origin, in the Zeiss 2D coordinates system, common to all planes. Unit: pixel (highest resolution level) + @CopyByRef + private List coreIndexToOy = new ArrayList<>(); + + // bio-formats core index the compression factor of the series. + @CopyByRef + private List coreIndexToCompression = new ArrayList<>(); + + // bio-formats core index the compression factor of the series. + @CopyByRef + private List coreIndexToSignature = new ArrayList<>(); + + // bio-formats core index the downscaling factor of the series. + @CopyByRef + private List coreIndexToDownscaleFactor = new ArrayList<>(); + + // Maps file part to the filename, in case of multipart file + @CopyByRef + private List filePartToFileName = new ArrayList<>(); // TODO: Find a way to not store the absolutepath + + @CopyByRef + private Map coreIndexToSeries = new HashMap<>(); + + // streamCurrentSeries is a temp field that should maybe be changed when setSeries is called + transient int streamCurrentPart = -1; + + // Core map structure for fast access to blocks: + // - first key: bio-formats core index + // - second key: czt index + @CopyByRef + private List< // CoreIndex + HashMap>> + coreIndexToTZCToMinimalBlocks = new ArrayList<>(); + + @CopyByRef + int nIlluminations, nRotations, nPhases; + + // ------------------------ METADATA FIELDS + @CopyByRef private MetadataStore store; - private HashMap pixels; - - private ArrayList segments; - private ArrayList planes; - private HashMap> indexIntoPlanes = - new HashMap>(); - private int rotations = 1; - private int positions = 1; - private int illuminations = 1; - private int acquisitions = 1; - private int mosaics = 1; - private int phases = 1; - private int angles = 1; - private int maxResolution = 0; - - private String imageName; - private String acquiredDate; - private String description; - private String userDisplayName, userName; - private String userFirstName, userLastName, userMiddleName; - private String userEmail; - private String userInstitution; - private String temperature, airPressure, humidity, co2Percent; - private String correctionCollar, medium, refractiveIndex; - private transient Time timeIncrement; - - private String zoom; - private String gain; - - private ArrayList channels = new ArrayList(); - private ArrayList binnings = new ArrayList(); - private ArrayList detectorRefs = new ArrayList(); - private ArrayList timestamps = new ArrayList(); - private transient ArrayList gains = new ArrayList(); - - private Length[] positionsX; - private Length[] positionsY; - private Length[] positionsZ; - - private int previousChannel = 0; - - private Boolean prestitched = null; - private String objectiveSettingsID; - private boolean hasDetectorSettings = false; - private int scanDim = 1; - - private String[] rotationLabels, phaseLabels, illuminationLabels; - - private transient DocumentBuilder parser; - - private ArrayList extraImages = new ArrayList(); - private int[] tileWidth; - private int[] tileHeight; - private int scaleFactor; - - private transient Length zStep; - - private transient int plateRows; - private transient int plateColumns; - private transient ArrayList platePositions = new ArrayList(); - private transient ArrayList fieldNames = new ArrayList(); - private transient ArrayList imageNames = new ArrayList(); + @CopyByRef + private ArrayList extraImages = new ArrayList<>(); + + @CopyByRef + int maxBlockSizeX = -1; + @CopyByRef + int maxBlockSizeY = -1; + + //----------------- CACHE + /** + * While the reader is not thread safe, the cache should be, because + * it is shared between multiple readers which can coexist in different threads + * if using the {@link ZeissCZIReader#copy()} method to duplicate the reader + */ + @CopyByRef + transient SubBlockLRUCache subBlockLRUCache = new SubBlockLRUCache(10 * 1024 * 1024, 400 * 1024 * 1024 ); + + @CopyByRef + transient Lock cacheLock = new ReentrantLock(); + + @CopyByRef + transient Set subBlocksCurrentlyLoading = new HashSet<>(); + + @CopyByRef + transient boolean useCache = true; // -- Constructor -- + final static String FORMAT = "Zeiss CZI (Quick Start)"; + final static String SUFFIX = "czi"; + /** Constructs a new Zeiss .czi reader. */ public ZeissCZIReader() { - super("Zeiss CZI", "czi"); + super(FORMAT, SUFFIX); domains = new String[] {FormatTools.LM_DOMAIN, FormatTools.HISTOLOGY_DOMAIN}; suffixSufficient = false; suffixNecessary = false; } - // -- IFormatReader API methods -- + /** Duplicates 'that' reader for parallel reading. + * Creating a reader with this constructor allows to keep a very low memory footprint + * because all immutable objects are re-used by reference. + * WARNING: calling {@link ZeissCZIReader#close()} on this or that reader will prevent the use + * of the other reader created with this constructor */ + public ZeissCZIReader(ZeissCZIReader that) { + super(FORMAT, SUFFIX); + domains = new String[] {FormatTools.LM_DOMAIN, FormatTools.HISTOLOGY_DOMAIN}; + suffixSufficient = false; + suffixNecessary = false; - /** - * @see loci.formats.IFormatReader#isThisType(RandomAccessInputStream) - */ - @Override - public boolean isThisType(RandomAccessInputStream stream) throws IOException { - final int blockLen = 10; - if (!FormatTools.validStream(stream, blockLen, true)) return false; - String check = stream.readString(blockLen); - return check.equals(CZI_MAGIC_STRING); - } + this.streamCurrentPart = -1; - /** - * @see loci.formats.IFormatReader#getSeriesUsedFiles(boolean) - */ - @Override - public String[] getSeriesUsedFiles(boolean noPixels) { - FormatTools.assertId(currentId, true, 1); - if (pixels == null || pixels.size() == 0 && noPixels) { - return null; - } - else if (noPixels) { - return null; - } - String[] files = new String[pixels.size() + 1]; - files[0] = currentId; - Integer[] keys = pixels.keySet().toArray(new Integer[pixels.size()]); - Arrays.sort(keys); - for (int i=0; i= channels.size()) - { - return null; + public void close(boolean fileOnly) throws IOException { + super.close(fileOnly); + if (!fileOnly) { + coreIndexToTZCToMinimalBlocks.clear(); // ZE big one! Hum, problem if another reader uses it... But ok + store = null; + // getStream().close(); done in the super method call + coreIndexToOx.clear(); + coreIndexToOy.clear(); + coreIndexToCompression.clear(); + coreIndexToSignature.clear(); + coreIndexToDownscaleFactor.clear(); + filePartToFileName.clear(); + coreIndexToSeries.clear(); + coreIndexToTZCToMinimalBlocks.clear(); // The big one! + extraImages.clear(); // Can be big as well + cacheLock.lock(); + subBlockLRUCache.clear(); + subBlocksCurrentlyLoading.clear(); + cacheLock.unlock(); } + } - byte[][] lut = new byte[3][256]; - - String color = channels.get(previousChannel).color; - if (color != null) { - color = color.replaceAll("#", ""); - try { - int colorValue = Integer.parseInt(color, 16); + /* @see loci.formats.FormatReader#initFile(String) */ + @Override + protected ArrayList getAvailableOptions() { + ArrayList optionsList = super.getAvailableOptions(); + optionsList.add(ALLOW_AUTOSTITCHING_KEY); + optionsList.add(INCLUDE_ATTACHMENTS_KEY); + optionsList.add(TRIM_DIMENSIONS_KEY); + optionsList.add(RELATIVE_POSITIONS_KEY); + return optionsList; + } - int redMax = (colorValue & 0xff0000) >> 16; - int greenMax = (colorValue & 0xff00) >> 8; - int blueMax = colorValue & 0xff; + // -- ZeissCZI-specific methods -- - for (int i=0; i= channels.size()) - { - return null; + public boolean canReadAttachments() { // TODO : handle this method + MetadataOptions options = getMetadataOptions(); + if (options instanceof DynamicMetadataOptions) { + return ((DynamicMetadataOptions) options).getBoolean( + INCLUDE_ATTACHMENTS_KEY, INCLUDE_ATTACHMENTS_DEFAULT); } + return INCLUDE_ATTACHMENTS_DEFAULT; + } - short[][] lut = new short[3][65536]; - - String color = channels.get(previousChannel).color; - if (color != null) { - color = color.replaceAll("#", ""); - try { - int colorValue = Integer.parseInt(color, 16); - - int redMax = (colorValue & 0xff0000) >> 16; - int greenMax = (colorValue & 0xff00) >> 8; - int blueMax = colorValue & 0xff; - - redMax = (int) (65535 * (redMax / 255.0)); - greenMax = (int) (65535 * (greenMax / 255.0)); - blueMax = (int) (65535 * (blueMax / 255.0)); - - for (int i=0; i 0) { - fill = (byte) 255; + private void swapRGBIfnecessary(byte[] buf, int compression, int bpp, int pixel) { + if (isRGB() /*&& !emptyTile*/ && compression != JPEGXR) { // TODO: case emptytile + // channels are stored in BGR order; red and blue channels need switching + // JPEG-XR data has already been reversed during decompression + int redOffset = bpp * 2; + int index = 0; + int nloops=buf.length/pixel; + for (int i=0; i it is already set when calling the method + + if ((useCache)&&(compression!=UNCOMPRESSED)) { + cacheLock.lock(); + if (subBlockLRUCache.containsKey(block)) { + byte[] bytes = subBlockLRUCache.get(block).get(); + if (bytes!=null) { + // - cache hit for block + subBlockLRUCache.touch(block, bytes); // Updates order + cacheLock.unlock(); + return bytes; + } + } + + // - block not in cache + if (subBlocksCurrentlyLoading.contains(block)) { + // There's already, in a different thread, a reader instance + // computing the same block. We need to wait for it to finish + try { + do { + // - waiting for block to be computed + cacheLock.unlock(); + synchronized (subBlocksCurrentlyLoading) { + subBlocksCurrentlyLoading.wait(); + } + cacheLock.lock(); + // - is the right block being computed ? + } while (subBlocksCurrentlyLoading.contains(block)); + // Yes -> the right block has been computed + } catch (InterruptedException e) { + throw new RuntimeException(e); + } + byte[] bytes = subBlockLRUCache.get(block).get(); + if (bytes!=null) { + // - the block has been computed + subBlockLRUCache.touch(block, bytes); // put on top, for LRU cache + cacheLock.unlock(); + return bytes; + } else { + // weird: the block is not there in the end... maybe it's been removed from the cache + synchronized (subBlocksCurrentlyLoading) { + subBlocksCurrentlyLoading.add(block); + } + } + } else { + // This thread will compute this block + synchronized (subBlocksCurrentlyLoading) { + subBlocksCurrentlyLoading.add(block); + } + } + cacheLock.unlock(); + } - if (isThumbnailSeries()) { - // thumbnail, label, or preview image stored as an attachment + LibCZI.SubBlockSegment subBlock = LibCZI.getBlock(s, block.filePosition); + long blockDataOffset = subBlock.dataOffset; + long blockDataSize = subBlock.data.dataSize; - int index = getCoreIndex() - (core.size() - extraImages.size()); - byte[] fullPlane = extraImages.get(index).attachmentData; - RandomAccessInputStream s = new RandomAccessInputStream(fullPlane); - try { - readPlane(s, x, y, w, h, buf); + s.seek(blockDataOffset); + + if (compression == UNCOMPRESSED) { + if (buf == null) { + buf = new byte[(int) blockDataSize]; } - finally { - s.close(); + if (tile != null) { + readPlane(s, tile.x, tile.y, tile.width, tile.height,0,storedSizeX,storedSizeY,buf); } + else { + s.readFully(buf); + } + swapRGBIfnecessary(buf, UNCOMPRESSED, bpp, totalBpp); return buf; } - previousChannel = getZCTCoords(no)[1]; - - int currentIndex = getCoreIndex(); - - Region image = new Region(x, y, w, h); - - int bpp = FormatTools.getBytesPerPixel(getPixelType()); - int pixel = getRGBChannelCount() * bpp; - int outputRowLen = w * pixel; - - int outputRow = 0, outputCol = 0; + byte[] data = new byte[(int) blockDataSize]; + s.read(data); - boolean validScanDim = - scanDim == (getImageCount() / (getSizeC() * phases)) && scanDim > 1; - if (planes.size() == getImageCount()) { - validScanDim = false; - } + int bytesPerPixel = FormatTools.getBytesPerPixel(getPixelType()); + CodecOptions options = new CodecOptions(); + options.interleaved = isInterleaved(); + options.littleEndian = isLittleEndian(); + options.bitsPerSample = bytesPerPixel * 8; + options.maxBytes = getSizeX() * getSizeY() * getRGBChannelCount() * bytesPerPixel; - Arrays.fill(buf, getFillColor()); - boolean emptyTile = true; - int compression = -1; - try { - int minTileX = Integer.MAX_VALUE, minTileY = Integer.MAX_VALUE; - int baseResolution = currentIndex; - while (baseResolution > 0 && core.get(baseResolution - 1).sizeX > core.get(baseResolution).sizeX) { - baseResolution--; - } - for (SubBlock plane : planes) { - if ((plane.planeIndex == no && ((maxResolution == 0 && plane.coreIndex == currentIndex) || - (maxResolution > 0 && plane.coreIndex == baseResolution))) || - (plane.planeIndex == previousChannel && validScanDim)) - { - if (plane.row < minTileY) { - minTileY = plane.row; + switch (compression) { + case JPEG: + data = new JPEGCodec().decompress(data, options); + break; + case LZW: + data = new LZWCodec().decompress(data, options); + break; + case JPEGXR: + options.width = storedSizeX; + options.height = storedSizeY; + options.maxBytes = options.width * options.height * + getRGBChannelCount() * bytesPerPixel; + try { + data = new JPEGXRCodec().decompress(data, options); + } + catch (FormatException e) { + if (data.length == options.maxBytes) { + logger.debug("Invalid JPEG-XR compression flag"); } - if (plane.col < minTileX) { - minTileX = plane.col; + else { + logger.warn("Could not decompress block; some pixels may be 0", e); + data = new byte[options.maxBytes]; } } - } - for (SubBlock plane : planes) { - if ((plane.coreIndex == currentIndex && plane.planeIndex == no) || - (plane.planeIndex == previousChannel && validScanDim)) - { - int res = (int) Math.pow(scaleFactor, plane.resolutionIndex); - - int realX = plane.x / res; - int realY = plane.y / res; - - if ((prestitched != null && prestitched) || validScanDim) { - Region tile = new Region(plane.col, plane.row, realX, realY); - if (validScanDim) { - tile.y += (no / getSizeC()); - image.height = scanDim; - } - if (prestitched != null && prestitched && realX == getSizeX() && realY == getSizeY()) { - tile.x = 0; - tile.y = 0; - } - else if (prestitched != null && prestitched) { - // normalize the coordinates such that minimum row/col values are 0 - tile.x -= minTileX; - tile.y -= minTileY; + break; + case ZSTD_0: + data = new ZstdCodec().decompress(data); + break; + case ZSTD_1: + boolean highLowUnpacking = false; + int pointer = 0; + try (RandomAccessInputStream stream = new RandomAccessInputStream(data)) { + int sizeOfHeader = readVarint(stream); + while (stream.getFilePointer() < sizeOfHeader) { + int chunkID = readVarint(stream); + // only one chunk ID defined so far + if (chunkID == 1) { + int payload = stream.read(); + highLowUnpacking = (payload & 1) == 1; + } else { + throw new FormatException("Invalid chunk ID: " + chunkID); } - tile.x /= res; - tile.y /= res; - - if (tile.intersects(image)) { - emptyTile = false; - compression = plane.directoryEntry.compression; - byte[] rawData = new SubBlock(plane).readPixelData(); - Region intersection = tile.intersection(image); - int intersectionX = 0; - - if (tile.x < image.x) { - intersectionX = image.x - tile.x; - } - - outputCol = (intersection.x - x) * pixel; - outputRow = intersection.y - y; - if (validScanDim) { - outputRow -= tile.y; - } - - if (rawData.length < realX * realY * pixel) { - realX = rawData.length / (realY * pixel); - } - else if (rawData.length == (realX + 1) * (realY + 1) * pixel) { - realX++; - realY++; - } + } + // safe cast because stream wraps a byte array + pointer = (int) stream.getFilePointer(); + } - int rowLen = pixel * (int) Math.min(intersection.width, realX); - int outputOffset = outputRow * outputRowLen + outputCol; - for (int trow=0; trow buf.length || pixels.size() > 0) { - RandomAccessInputStream s = new RandomAccessInputStream(rawData); - try { - readPlane(s, x, y, w, h, realX - getSizeX(), buf); - emptyTile = false; - } - finally { - s.close(); - } - } - else { - emptyTile = false; - } - break; + } + else { + logger.warn("ZSTD-1 compression used, but no high/low byte unpacking"); + data = decoded; + } + + break; + case 104: // camera-specific packed pixels + data = decode12BitCamera(data, options.maxBytes); + // reverse column ordering + for (int row=0; row= data.length) { + System.arraycopy(data, 0, buf, 0, data.length); + swapRGBIfnecessary(buf, UNCOMPRESSED, bpp, totalBpp); + if (useCache) { + cacheLock.lock(); + // Block just computed + subBlockLRUCache.touch(block, buf); + // Put in cache + subBlocksCurrentlyLoading.remove(block); + cacheLock.unlock(); + synchronized (subBlocksCurrentlyLoading) { + subBlocksCurrentlyLoading.notifyAll(); // Wake the threads waiting for this block } } + return buf; + } + swapRGBIfnecessary(data, UNCOMPRESSED, bpp, totalBpp); + if (useCache) { + cacheLock.lock(); + subBlockLRUCache.touch(block, data); + subBlocksCurrentlyLoading.remove(block); + cacheLock.unlock(); + synchronized (subBlocksCurrentlyLoading) { + subBlocksCurrentlyLoading.notifyAll(); + } } + return data; + } - return buf; + private static int readVarint(RandomAccessInputStream stream) throws IOException { + byte a = stream.readByte(); + // if high bit set, read next byte + // at most 3 bytes read + if ((a & 0x80) == 0x80) { + byte b = stream.readByte(); + if ((b & 0x80) == 0x80) { + byte c = stream.readByte(); + return (c << 14) | ((b & 0x7f) << 7) | (a & 0x7f); + } + return (b << 7) | (a & 0x7f); + } + return a & 0xff; + } + + private static byte[] decode12BitCamera(byte[] data, int maxBytes) throws IOException { + byte[] decoded = new byte[maxBytes]; + + RandomAccessInputStream bb = new RandomAccessInputStream( + new ByteArrayHandle(data)); + byte[] fourBits = new byte[(maxBytes / 2) * 3]; + int pt = 0; + while (pt < fourBits.length) { + fourBits[pt++] = (byte) bb.readBits(4); + } + bb.close(); + for (int index=0; index it's preferable to avoid decompressing multiple times the same block + // but with overlapping, anyway, and without caching, multiple decompression of the same block is hard to avoid @Override public int getOptimalTileWidth() { - if (maxResolution > 0 && getCoreIndex() < core.size() - extraImages.size()) { - return (int) Math.min(1024, getSizeX()); - } - if (tileWidth != null && getCoreIndex() < tileWidth.length) { - int width = tileWidth[getCoreIndex()]; - if (width == 0 && getCoreIndex() > 0) { - width = tileWidth[getCoreIndex() - 1] / 2; - } - return width == 0 ? 1024 : width; + if (maxBlockSizeX>0) { + return Math.min(2048, maxBlockSizeX); + } else { + return Math.min(2048, getSizeX()); } - return super.getOptimalTileWidth(); } /* @see loci.formats.IFormatReader#getOptimalTileHeight() */ @Override public int getOptimalTileHeight() { - if (maxResolution > 0 && getCoreIndex() < core.size() - extraImages.size()) { - return (int) Math.min(1024, getSizeY()); - } - if (tileHeight != null && getCoreIndex() < tileHeight.length) { - int height = tileHeight[getCoreIndex()]; - if (height == 0 && getCoreIndex() > 0) { - height = tileHeight[getCoreIndex() - 1] / 2; - } - return height == 0 ? 1024 : height; + if (maxBlockSizeY>0) { + return Math.min(2048, maxBlockSizeY); + } else { + return Math.min(2048, getSizeY()); } - return super.getOptimalTileHeight(); } - // -- Internal FormatReader API methods -- - - /* @see loci.formats.FormatReader#initFile(String) */ @Override - protected ArrayList getAvailableOptions() { - ArrayList optionsList = super.getAvailableOptions(); - optionsList.add(ALLOW_AUTOSTITCHING_KEY); - optionsList.add(INCLUDE_ATTACHMENTS_KEY); - optionsList.add(TRIM_DIMENSIONS_KEY); - optionsList.add(RELATIVE_POSITIONS_KEY); - return optionsList; + public byte[] openBytes(int no, byte[] buf, int x, int y, int w, int h) throws FormatException, IOException { + + FormatTools.checkPlaneParameters(this, no, buf.length, x, y, w, h); + + if (isThumbnailSeries()) { + // thumbnail, label, or preview image stored as an attachment + int index = getCoreIndex() - (core.size() - extraImages.size()); + byte[] fullPlane = extraImages.get(index); + try (RandomAccessInputStream s = new RandomAccessInputStream(fullPlane)) { + readPlane(s, x, y, w, h, buf); + } + return buf; + } + + int currentIndex = getCoreIndex(); + int bpp = FormatTools.getBytesPerPixel(getPixelType()); + int nCh = getRGBChannelCount(); + int bytesPerPixel = (isRGB()?nCh:1) * bpp; + int baseResolution = currentIndex; + + Region image = new Region(x, y, w, h); + + // Because series are sorted along their resolution level, that's a way to find which + // resolution level is the lowest one - but that looks very brittle - what if you have a very + // while the downscaling is decreasing, let's decrement baseresolution + // what this assumes is that the resolution level whereby downscaling = 1 is always present + + int[] czt = this.getZCTCoords(no); + CZTKey key = new CZTKey(czt[1], czt[0], czt[2]); + + while (baseResolution > 0 && + coreIndexToDownscaleFactor.get(baseResolution) > + coreIndexToDownscaleFactor.get(baseResolution-1)) { + baseResolution--; + } + + // The data is somewhere in these blocks + List blocks = coreIndexToTZCToMinimalBlocks.get(currentIndex).get(key); + + if (blocks == null) return buf; // No block found -> empty image. TODO : black or white ? + + for (MinDimEntry block : blocks) { + Region blockRegion = new Region( + block.dimensionStartX/ coreIndexToDownscaleFactor.get(coreIndex) - coreIndexToOx.get(coreIndex), + block.dimensionStartY/ coreIndexToDownscaleFactor.get(coreIndex) - coreIndexToOy.get(coreIndex), + block.storedSizeX, + block.storedSizeY + ); + + if (image.intersects(blockRegion)) { + RandomAccessInputStream stream = getStream(block.filePart); + + if (image.equals(blockRegion)) { + // Best case scenario + return readRawPixelData( + block, + coreIndexToCompression.get(coreIndex), + block.storedSizeX, + block.storedSizeY, + stream,null,buf, bpp, bytesPerPixel); + } else { + // We need to copy, taking in consideration the size taken by the image + // We can potentially crop what's read + + int compression = coreIndexToCompression.get(coreIndex); + + Region regionRead; + // If the data is uncompressed, we can skip reading some data + if (compression == UNCOMPRESSED) { + regionRead = image.intersection(blockRegion); + } else { + regionRead = blockRegion; + } + + Region tileInBlock = new Region(regionRead.x-blockRegion.x, regionRead.y-blockRegion.y, regionRead.width, regionRead.height); + + byte[] rawData = readRawPixelData( + block, + compression, + block.storedSizeX, + block.storedSizeY, + stream, compression==UNCOMPRESSED? tileInBlock: null, // can't really optimize with compressed block + compression==UNCOMPRESSED? DataTools.allocate(tileInBlock.width, tileInBlock.height, nCh, bpp): null, + bpp, bytesPerPixel); + + // We need to basically crop a rectangle with a rectangle, of potentially different sizes + // Let's find out the position of the block in the image referential + int blockOriX = regionRead.x-image.x; + int skipBytesStartX = 0; + int skipBytesBufStartX = 0; + if (blockOriX<0) { + skipBytesStartX = -blockOriX*bytesPerPixel; + } else { + skipBytesBufStartX = blockOriX*bytesPerPixel; + } + int blockEndX = (regionRead.x+regionRead.width)-(image.x+image.width); + int skipBytesEndX = 0; + if (blockEndX>0) { + skipBytesEndX = blockEndX*bytesPerPixel; + } + int nBytesToCopyPerLine = (regionRead.width*bytesPerPixel-skipBytesStartX-skipBytesEndX); + int blockOriY = regionRead.y-image.y; + int skipLinesRawDataStart = 0; + int skipLinesBufStart = 0; + if (blockOriY<0) { + skipLinesRawDataStart = -blockOriY; + } else { + skipLinesBufStart = blockOriY; + } + int blockEndY = (regionRead.y+regionRead.height)-(image.y+image.height); + int skipLinesEnd = Math.max(blockEndY, 0); + int totalLines = regionRead.height-skipLinesRawDataStart-skipLinesEnd; + int nBytesPerLineRawData = regionRead.width*bytesPerPixel; + int nBytesPerLineBuf = image.width*bytesPerPixel; + int offsetRawData = skipLinesRawDataStart*nBytesPerLineRawData+skipBytesStartX; + int offsetBuf = skipLinesBufStart*nBytesPerLineBuf+skipBytesBufStartX; + + for (int i=0; i getAvailableOptions() { protected void initFile(String id) throws FormatException, IOException { super.initFile(id); - parser = XMLTools.createBuilder(); - - // switch to the master file if this is part of a multi-file dataset + // Switch to the master file if this is part of a multi-file dataset int lastDot = id.lastIndexOf("."); String base = lastDot < 0 ? id : id.substring(0, lastDot); if (base.endsWith(")") && isGroupFiles()) { @@ -658,17 +942,11 @@ protected void initFile(String id) throws FormatException, IOException { } } - CoreMetadata ms0 = core.get(0); + // At this point of the initFile method, it's guaranteed that id is the id of the master file + store = makeFilterMetadata(); // For metadata + core.get(0).littleEndian = true; // We assume that CZI files are always little endian. Setting the value at this point in the method ensures that isLittleEndian() calls return true - ms0.littleEndian = true; - - pixels = new HashMap(); - segments = new ArrayList(); - planes = new ArrayList(); - - readSegments(id); - - // check if we have the master file in a multi-file dataset + // Multiple czi files may exist // file names are not stored in the files; we have to rely on a // specific naming convention: // @@ -687,1696 +965,1566 @@ protected void initFile(String id) throws FormatException, IOException { base = base.substring(0, lastDot); } + // Map file part and CZI segments found in the file + // but only the segments needed for the file initialisation... + Map cziPartToSegments = new HashMap<>(); + cziPartToSegments.put(0, new CZISegments(id, isLittleEndian())); // CZISegments constructor parses CZI file segments + filePartToFileName.add(id); + // And we the additional parts. Location parent = file.getParentFile(); String[] list = parent.list(true); for (String f : list) { if (f.startsWith(base + "(") || f.startsWith(base + " (")) { String part = f.substring(f.lastIndexOf("(") + 1, f.lastIndexOf(")")); try { - pixels.put(Integer.parseInt(part), - new Location(parent, f).getAbsolutePath()); + String filePartPath = new Location(parent, f).getAbsolutePath(); + cziPartToSegments.put(Integer.parseInt(part), + new CZISegments(filePartPath, isLittleEndian())); + filePartToFileName.add(filePartPath); } catch (NumberFormatException e) { LOGGER.debug("{} not included in multi-file dataset", f); } } } - Integer[] keys = pixels.keySet().toArray(new Integer[pixels.size()]); - Arrays.sort(keys); - for (Integer key : keys) { - readSegments(pixels.get(key)); - } - - calculateDimensions(); - - if (planes.size() == 0) { - throw new FormatException( - "Pixel data could not be found; this file may be corrupted"); - } - - int firstX = planes.get(0).x; - int firstY = planes.get(0).y; + // How many dimensions exist for CZI ? A lot + //Z The Z-dimension. + //C The C-dimension ("channel"). + //T The T-dimension ("time"). + //R The R-dimension ("rotation"). + //S The S-dimension ("scene"). + //I The I-dimension ("illumination"). + //H The H-dimension ("phase"). + //V The V-dimension ("view"). + //B The B-dimension ("block") - its use is deprecated. + //M The M-dimension ("mosaic") -> why there's no trace of it in libczi ??? + + // Then, we add two extra dimensions for convenience: + // PY, which identifies the pyramidal level (RESOLUTION_LEVEL_DIMENSION) + // PA, which identifies the file part (FILE_PART_DIMENSION) + + // To build a series, one should: + // - Split according to view, phase, illumination, rotation, scene, and mosaic ? + // - Merge according to Z, T + // - Merge according to C, except if pixels types are different (unimplemented, case not found in tested images. But is it possible?) + + // Adding the extra dimension in each subblock : part, and resolution level + // For the series order, each dimension has a priority, set by the method dimensionPriority + + // What do I want to do ? + // I want to find the number of series. For that, I make a unique signature + // of each subblock with its dimension signature, then count the number of + // signature in the signature set. + + // The signature alphabetical order will be used for the ordering of the series + // Here's some example signatures: + // H0S04PY00001 + // H0S03PY00001 + // H0S02PY00001 -> phase 0, scene 2, pyramidal level 1 (highest resolution) + // H0S01PY00001 + // H0S00PY00001 + // PY is always last, that's how bio-formats work: resolution levels of a series always follow each other (core indices) + + Map maxValuePerDimension = new HashMap<>(); + + // Then we look at the max value in each dimension, to know how many digits are needed to write the signature + // and proper alphabetical ordering + cziPartToSegments.forEach((part, cziSegments) -> { // For each part + Arrays.asList(cziSegments.subBlockDirectory.data.entries).forEach( // and each entry + entry -> { + for (LibCZI.SubBlockSegment.SubBlockSegmentData.SubBlockDirectoryEntryDV.DimensionEntry dimEntry: entry.getDimensionEntries()) { + //int nDigits = String.valueOf(dimEntry.start).length(); // TODO: Can this be negative ? + int val = dimEntry.start; + if (!maxValuePerDimension.containsKey(dimEntry.dimension)) { + maxValuePerDimension.put(dimEntry.dimension, dimEntry.start); + } else { + int curMax = maxValuePerDimension.get(dimEntry.dimension); + if (val>curMax) { + maxValuePerDimension.put(dimEntry.dimension, val); + } + } + } + } + ); + }); - if (getSizeC() == 0) { - ms0.sizeC = 1; - } - if (getSizeZ() == 0) { - ms0.sizeZ = 1; - } - if (getSizeT() == 0) { - ms0.sizeT = 1; - } - if (getImageCount() == 0) { - ms0.imageCount = ms0.sizeZ * ms0.sizeT; - } + nIlluminations = maxValuePerDimension.containsKey("I")? maxValuePerDimension.get("I")+1:1; - int originalC = getSizeC(); - convertPixelType(planes.get(0).directoryEntry.pixelType); + nRotations = maxValuePerDimension.containsKey("R")? maxValuePerDimension.get("R")+1:1; - // remove any invalid SubBlocks + nPhases = maxValuePerDimension.containsKey("H")? maxValuePerDimension.get("H")+1:1; - int bpp = FormatTools.getBytesPerPixel(getPixelType()); - if (isRGB()) { - bpp *= (getSizeC() / originalC); - } - int fullResBlockCount = planes.size(); - for (int i=0; i= Integer.MAX_VALUE || size < 0) { - // check for reduced resolution in the pyramid - DimensionEntry[] entries = planes.get(i).directoryEntry.dimensionEntries; - int pyramidType = planes.get(i).directoryEntry.pyramidType; - if ((pyramidType == 1 || pyramidType == 2 || compression == JPEGXR) && - (compression == JPEGXR || size == entries[0].storedSize * entries[1].storedSize * bpp)) - { - int scale = planes.get(i).x / entries[0].storedSize; - if (scale == 1 || (((scale % 2) == 0 || (scale % 3) == 0) && allowAutostitching())) { - if (scale > 1 && scaleFactor == 0) { - scaleFactor = scale % 2 == 0 ? 2 : 3; - } - planes.get(i).coreIndex = 0; - while (scale > 1) { - scale /= scaleFactor; - planes.get(i).coreIndex++; - } - if (planes.get(i).coreIndex > maxResolution) { - maxResolution = planes.get(i).coreIndex; - } - } - else { - LOGGER.trace( - "removing block #{}; calculated size = {}, recorded size = {}, scale = {}", - i, planeSize, size, scale); - planes.remove(i); - i--; - } - } - else { - LOGGER.trace( - "removing block #{}; calculated size = {}, recorded size = {}", - i, planeSize, size); - planes.remove(i); - i--; - } - fullResBlockCount--; - } - else { - scanDim = (int) (size / planeSize); - } - } - else { - byte[] pixels = planes.get(i).readPixelData(); - if (pixels.length < planeSize || planeSize >= Integer.MAX_VALUE) { - LOGGER.trace( - "removing block #{}; calculated size = {}, decoded size = {}", - i, planeSize, pixels.length); - planes.remove(i); - i--; - } - else { - scanDim = (int) (pixels.length / planeSize); - } - } - } + int nChannels = maxValuePerDimension.containsKey("C")? maxValuePerDimension.get("C")+1:1; - if (getSizeZ() == 0) { - ms0.sizeZ = 1; - } - if (getSizeC() == 0) { - ms0.sizeC = 1; - } - if (getSizeT() == 0) { - ms0.sizeT = 1; - } + int nSlices = maxValuePerDimension.containsKey("Z")? maxValuePerDimension.get("Z")+1:1; - // set modulo annotations - // rotations -> modulo Z - // illuminations -> modulo C - // phases -> modulo T - - LOGGER.trace("rotations = {}", rotations); - LOGGER.trace("illuminations = {}", illuminations); - LOGGER.trace("phases = {}", phases); - - LOGGER.trace("positions = {}", positions); - LOGGER.trace("acquisitions = {}", acquisitions); - LOGGER.trace("mosaics = {}", mosaics); - LOGGER.trace("angles = {}", angles); - - ms0.moduloZ.step = ms0.sizeZ; - ms0.moduloZ.end = ms0.sizeZ * (rotations - 1); - ms0.moduloZ.type = FormatTools.ROTATION; - ms0.sizeZ *= rotations; - - ms0.moduloC.step = ms0.sizeC; - ms0.moduloC.end = ms0.sizeC * (illuminations - 1); - ms0.moduloC.type = FormatTools.ILLUMINATION; - ms0.moduloC.parentType = FormatTools.CHANNEL; - ms0.sizeC *= illuminations; - - ms0.moduloT.step = ms0.sizeT; - ms0.moduloT.end = ms0.sizeT * (phases - 1); - ms0.moduloT.type = FormatTools.PHASE; - ms0.sizeT *= phases; - - // finish populating the core metadata - - int seriesCount = positions * acquisitions * angles; - int originalMosaicCount = mosaics; - if (maxResolution == 0) { - seriesCount *= mosaics; - } - else { - prestitched = true; - } + int nFrames = maxValuePerDimension.containsKey("T")? maxValuePerDimension.get("T")+1:1; - ms0.imageCount = getSizeZ() * (isRGB() ? getSizeC()/3 : getSizeC()) * getSizeT(); - - LOGGER.trace("Size Z = {}", getSizeZ()); - LOGGER.trace("Size C = {}", getSizeC()); - LOGGER.trace("Size T = {}", getSizeT()); - LOGGER.trace("is RGB = {}", isRGB()); - LOGGER.trace("calculated image count = {}", ms0.imageCount); - LOGGER.trace("number of available planes = {}", planes.size()); - LOGGER.trace("prestitched = {}", prestitched); - LOGGER.trace("scanDim = {}", scanDim); - - int calculatedSeries = fullResBlockCount / getImageCount(); - if (((mosaics == seriesCount) || (positions == seriesCount)) && - ((seriesCount == calculatedSeries) || - (maxResolution > 0 && seriesCount * mosaics == calculatedSeries)) && - prestitched != null && prestitched) - { - boolean equalTiles = true; - for (SubBlock plane : planes) { - if (plane.x != planes.get(0).x || plane.y != planes.get(0).y) { - equalTiles = false; + Map maxDigitPerDimension = new HashMap<>(); + maxValuePerDimension.keySet().forEach(dim -> { + switch (dim) { + case "C": + // illuminations -> modulo C + maxDigitPerDimension.put(dim, String.valueOf(maxValuePerDimension.get(dim)*nIlluminations).length()); break; - } - } - if ((getSizeX() > planes.get(0).x || - (getSizeX() == planes.get(0).x && - calculatedSeries == seriesCount * mosaics * positions)) && !equalTiles && allowAutostitching()) - { - // image was fused; treat the mosaics as a single image - seriesCount = 1; - positions = 1; - acquisitions = 1; - mosaics = 1; - angles = 1; - } - else { - SubBlock lastPlane = planes.get(planes.size() - 1); - int newX = lastPlane.x; - int newY = lastPlane.y; - if (allowAutostitching() && (ms0.sizeX < newX || ms0.sizeY < newY)) { - prestitched = true; - if (maxResolution > 0) { - mosaics = 1; - } - } - else { - prestitched = true; - } - - // don't shrink the dimensions if prestitching is allowed - // implies that the image size is being set to a tile size - DimensionEntry mosaicDimension = lastPlane.directoryEntry.getDimensionEntry("M"); - if (!prestitched || (mosaicDimension != null && mosaicDimension.start == 0)) { - ms0.sizeX = newX; - ms0.sizeY = newY; - } - } - } - else if (!allowAutostitching() && calculatedSeries > seriesCount) { - ms0.sizeX = firstX; - ms0.sizeY = firstY; - prestitched = true; - } - else if (allowAutostitching() && prestitched == null && mosaics > 1) { - prestitched = (mosaics == seriesCount && mosaics == calculatedSeries) || - (mosaics == (seriesCount / positions)); - } - - if (ms0.imageCount * seriesCount > planes.size() * scanDim && - planes.size() > 0) - { - if (planes.size() != ms0.imageCount && planes.size() != ms0.sizeT && - (planes.size() % (seriesCount * getSizeZ())) == 0) - { - if (!isGroupFiles() && planes.size() == (ms0.imageCount * seriesCount) / positions) { - seriesCount /= positions; - positions = 1; - } - } - else if (planes.size() == ms0.sizeT || planes.size() == ms0.imageCount || - (!isGroupFiles() && positions > 1)) - { - positions = 1; - acquisitions = 1; - mosaics = 1; - angles = 1; - seriesCount = 1; - } - else if (seriesCount > mosaics && mosaics > 1 && prestitched != null && prestitched) { - seriesCount /= mosaics; - mosaics = 1; - } - } + case "Z": + // rotations -> modulo Z + maxDigitPerDimension.put(dim, String.valueOf(maxValuePerDimension.get(dim)*nRotations).length()); + break; + case "T": + // phases -> modulo T + maxDigitPerDimension.put(dim, String.valueOf(maxValuePerDimension.get(dim)*nPhases).length()); + break; + default: + maxDigitPerDimension.put(dim, String.valueOf(maxValuePerDimension.get(dim)).length()); + } + }); + + // Ready to build the signature + Map> coreSignatureToBlocks = new HashMap<>(); + maxDigitPerDimension.put(RESOLUTION_LEVEL_DIMENSION,5); // Let's hope that the downsampling ratio never exceeds 9999 TODO : improve + maxDigitPerDimension.put(FILE_PART_DIMENSION, String.valueOf(cziPartToSegments.size()).length()); + + // Write all signatures + cziPartToSegments.forEach((part, cziSegments) -> { // For each part + Arrays.asList(cziSegments.subBlockDirectory.data.entries).forEach( // and each entry + entry -> { + int downscalingFactor = entry.getDimension("X").size/entry.getDimension("X").storedSize; + if ((downscalingFactor==1)||(allowAutostitching())) { + // Split by resolution level if flattenedResolutions is true + ModuloDimensionEntries moduloEntry = new ModuloDimensionEntries(entry, + nRotations, nIlluminations, nPhases, + nChannels, nSlices, nFrames, part); + + CoreSignature coreSignature = new CoreSignature(moduloEntry + , RESOLUTION_LEVEL_DIMENSION, + downscalingFactor,//getDownSampling(entry), + maxDigitPerDimension::get, + allowAutostitching(), + FILE_PART_DIMENSION, part); + if (!coreSignatureToBlocks.containsKey(coreSignature)) { + coreSignatureToBlocks.put(coreSignature, new ArrayList<>()); + } + coreSignatureToBlocks.get(coreSignature).add(moduloEntry); + } + }); + }); + + // Sort them + List orderedCoreSignatureList = coreSignatureToBlocks.keySet().stream().sorted().collect(Collectors.toList()); + + // We now know how many core index are present in the image... except for extra images! + + core = new ArrayList<>(); + int idxCoreResolutionLevelStart = -1; + int idxSeries = -1; + // These variables (idxCoreResolutionLevelStart and idxSeries) roles are to keep track of the previous highest + // resolution level, and thus to abide by the rule nPixres0 = nPixres_i * downsampling_i + int previousMinX_maxRes = -1, previousMinY_maxRes = -1, previousPixX_maxRes = -1, previousPixY_maxRes = -1; + for (int iCore = 0; iCore modulo Z + // illuminations -> modulo C + // phases -> modulo T + + core_i.moduloZ.step = nSlices; + core_i.moduloZ.end = nSlices * (nRotations - 1); + core_i.moduloZ.type = FormatTools.ROTATION; + + core_i.moduloC.step = nChannels; + core_i.moduloC.end = nChannels * (nIlluminations - 1); + core_i.moduloC.type = FormatTools.ILLUMINATION; + core_i.moduloC.parentType = FormatTools.CHANNEL; + + core_i.moduloT.step = nFrames; + core_i.moduloT.end = nFrames * (nPhases - 1); + core_i.moduloT.type = FormatTools.PHASE; + + //--------------- END OF MODULO + + core.add(core_i); + core_i.orderCertain = true; + core_i.dimensionOrder = "XYCZT"; + core_i.littleEndian = true; + CoreSignature coreSignature = orderedCoreSignatureList.get(iCore); + ModuloDimensionEntries model = coreSignatureToBlocks.get(coreSignature).get(0); + int[] coordsOrigin = setOriginAndSize(core_i, + coreSignatureToBlocks.get(coreSignature), + previousMinX_maxRes, + previousMinY_maxRes, + previousPixX_maxRes, + previousPixY_maxRes); + if (model.downSampling == 1) { + previousMinX_maxRes = coordsOrigin[0]; + previousMinY_maxRes = coordsOrigin[1]; + previousPixX_maxRes = core_i.sizeX; + previousPixY_maxRes = core_i.sizeY; + } + convertPixelType(core_i, model.getPixelType()); + coreIndexToCompression.add(model.getCompression()); + coreIndex = iCore; + coreIndexToDownscaleFactor.add(model.getDownSampling()); + + coreIndexToOx.add(coordsOrigin[0]); + coreIndexToOy.add(coordsOrigin[1]); + core_i.bitsPerPixel = + core_i.imageCount = (core_i.rgb ? core_i.sizeC/3 : core_i.sizeC)*core_i.sizeT*core_i.sizeZ; + + // We assert that all sub blocks do have the same size pixel type + // Series are ordered by dimension changes, and the (non-ignored) dimension + // with the highest priority is the resolution level + // So the downsampling will go : 1, 2, 4, 8, 1 (change!), 2, 4, 8, 1 (change), 2, 4, 1 (change), 1 (change) + // When a change is noticed, all previous core indices belong to the same series (unless flat resolution == false. + // see FormatReader#getSeriesToCoreIndex + + if (model.getDownSampling()==1) { + if (idxCoreResolutionLevelStart==-1) { + idxCoreResolutionLevelStart = iCore; + if (!hasFlattenedResolutions()) idxSeries++; + } else { + for (int j = idxCoreResolutionLevelStart; j not read for the sake of backward compatibility + + List sortedFileParts = cziPartToSegments.keySet().stream().sorted().collect(Collectors.toList()); - ms0.dimensionOrder = "XYCZT"; + try { + addLabelIfExists(sortedFileParts, cziPartToSegments, id);//, allPositionsInformation); + addSlidePreviewIfExists(sortedFileParts, cziPartToSegments, id);//, allPositionsInformation); + //getJPGThumbnailIfExists(sortedFileParts, cziPartToSegments, id); //disabled for bwd compatibility + } catch (DependencyException | ServiceException e) { + throw new RuntimeException(e); + } + + LOGGER.trace("#CoreSeries = {}", core.size()); + + // Logs all series info + for (int i = 0; i returns a list of blocks which + // differs only by X and Y coordinates. + // This structure will be trimmed down in size and stored in the mapCoreTZCToMinimalBlocks field + List< // CoreIndex + HashMap>> + List>> + mapCoreTZCToBlocks = new ArrayList<>(); + + for (int iCoreIndex = 0; iCoreIndex()); + coreIndexToTZCToMinimalBlocks.add(iCoreIndex, new HashMap<>()); + HashMap> blocksInCore = mapCoreTZCToBlocks.get(iCoreIndex); + HashMap> minimalBlocksInCore = coreIndexToTZCToMinimalBlocks.get(iCoreIndex); + for (ModuloDimensionEntries block: coreSignatureToBlocks.get(coreSignature)) { + int c = block.getDimension("C").start; + int z = (block.hasDimension("Z"))? block.getDimension("Z").start: 0; + int t = (block.hasDimension("T"))? block.getDimension("T").start: 0; + CZTKey k = new CZTKey(c,z,t); + if (!minimalBlocksInCore.containsKey(k)) { + blocksInCore.put(k, new ArrayList<>()); + minimalBlocksInCore.put(k, new ArrayList<>()); + } + MinDimEntry mde = new MinDimEntry(block); // Makes a trimmed down version of the block in order to reduce the reader memory footprint getMinimalEntry(block);// + blocksInCore.get(k).add(mde); + minimalBlocksInCore.get(k).add(mde); + } + //In the end, there are 'blocksInCore.values().size()' blocks in the core 'iCoreIndex' + } + + // Initialize the reader store, and basically all metadata + new MetadataInitializer(this).initializeMetadata(cziPartToSegments, mapCoreTZCToBlocks); - ArrayList pixelTypes = new ArrayList(); - pixelTypes.add(planes.get(0).directoryEntry.pixelType); - if (maxResolution == 0) { - for (SubBlock plane : planes) { - if (!pixelTypes.contains(plane.directoryEntry.pixelType)) { - pixelTypes.add(plane.directoryEntry.pixelType); - } - plane.pixelTypeIndex = pixelTypes.indexOf(plane.directoryEntry.pixelType); - } + } - if (core.size() * pixelTypes.size() > 1) { - core.clear(); - for (int j=0; j 1) { - int newC = originalC / pixelTypes.size(); - add.sizeC = newC; - add.imageCount = add.sizeZ * add.sizeT; - add.rgb = false; - convertPixelType(add, pixelTypes.get(j)); - } - core.add(add); - } + private void addLabelIfExists(List sortedFileParts, Map cziPartToSegments, String id) throws IOException, FormatException, DependencyException, ServiceException {//}, AllPositionsInformation allPositionsInformation) throws IOException, FormatException, DependencyException, ServiceException { + for (int filePart: sortedFileParts) { + byte[] bytes = LibCZI.getLabelBytes(cziPartToSegments.get(filePart).attachmentDirectory, id, BUFFER_SIZE, isLittleEndian()); + if (bytes!=null) { + int nSeries = getSeriesCount(); + ServiceFactory factory = new ServiceFactory(); + OMEXMLService service = factory.getInstance(OMEXMLService.class); + OMEXMLMetadata omeXML = service.createOMEXMLMetadata(); + ZeissCZIReader labelReader = new ZeissCZIReader(); + String placeHolderName = "label.czi"; + // thumbReader.setMetadataOptions(getMetadataOptions()); + ByteArrayHandle stream = new ByteArrayHandle(bytes); + Location.mapFile(placeHolderName, stream); + labelReader.setMetadataStore(omeXML); + labelReader.setId(placeHolderName); + + CoreMetadata c = labelReader.getCoreMetadataList().get(0); + + if (c.sizeZ > 1 || c.sizeT > 1) { + return; } - } - } - - // usually this indicates a big image for which a pyramid is - // expected but not present - if ((prestitched != null && prestitched) && - seriesCount == (mosaics * positions) && maxResolution == 0) - { - seriesCount = positions; - } + core.add(new CoreMetadata(c)); + core.get(core.size() - 1).thumbnail = true; + extraImages.add(labelReader.openBytes(0)); + stream.close(); + coreIndexToSeries.put(core.size() - 1, nSeries); + Location.mapFile(placeHolderName, null); - if (seriesCount > 1 || maxResolution > 0) { - core.clear(); - for (int i=0; i 0 || (mosaics > 1 && seriesCount == positions) || - (mosaics == 1 && seriesCount > 1)) - { - tileWidth = new int[core.size()]; - tileHeight = new int[core.size()]; - for (int s=0; s 0) { - core.get(s).sizeX = 0; - core.get(s).sizeY = 0; - calculateDimensions(s, true); - } - if (originalMosaicCount > 1) { - // calculate total stitched size if the image was not fused - int minRow = Integer.MAX_VALUE; - int maxRow = Integer.MIN_VALUE; - int minCol = Integer.MAX_VALUE; - int maxCol = Integer.MIN_VALUE; - int x = 0, y = 0; - int lastX = 0, lastY = 0; - for (SubBlock plane : planes) { - if (plane.coreIndex != s) { - continue; - } - if (x == 0 && y == 0) { - x = plane.x; - y = plane.y; - } - if (plane.row < minRow) { - minRow = plane.row; - } - if (plane.row > maxRow) { - maxRow = plane.row; - } - if (plane.col < minCol) { - minCol = plane.col; - } - if (plane.col > maxCol) { - maxCol = plane.col; - } - if (plane.x > tileWidth[s]) { - tileWidth[s] = plane.x; - } - if (plane.y > tileHeight[s]) { - tileHeight[s] = plane.y; - } - if (plane.row == maxRow && plane.col == maxCol) { - lastX = plane.x; - lastY = plane.y; - } - } - - // don't overwrite the dimensions if stitching already occurred - if (core.get(s).sizeX == x && core.get(s).sizeY == y) { - core.get(s).sizeX = (lastX + maxCol) - minCol; - core.get(s).sizeY = (lastY + maxRow) - minRow; - } - } - boolean keepMissingPyramid = false; - for (int r=0; r 0 && !keepMissingPyramid) { - core.remove(s + r); - core.get(s).resolutionCount--; - // adjust the core indexes of any subsequent planes - for (SubBlock plane : planes) { - if (plane.coreIndex > s + r) { - plane.coreIndex--; - } - } - r--; - } - else { - int div = (int) Math.pow(scaleFactor, r); - if (r == 0 && s > 0 && core.get(s).sizeX == 1) { - core.get(s).sizeX = core.get(s - maxResolution).sizeX; - core.get(s).sizeY = core.get(s - maxResolution).sizeY; - } - else { - core.get(s + r).sizeX = core.get(s).sizeX / div; - core.get(s + r).sizeY = core.get(s).sizeY / div; - } - tileWidth[s + r] = tileWidth[s] / div; - tileHeight[s + r] = tileHeight[s] / div; - } + } - if (r == 0 && !hasValidPlane) { - keepMissingPyramid = true; - } - } - if (trimDimensions()) { - calculateDimensions(s, true); + private void addSlidePreviewIfExists(List sortedFileParts, Map cziPartToSegments, String id) throws IOException, FormatException, DependencyException, ServiceException {//, AllPositionsInformation allPositionsInformation) throws IOException, FormatException, DependencyException, ServiceException { + for (int filePart: sortedFileParts) { + byte[] bytes = LibCZI.getPreviewBytes(cziPartToSegments.get(filePart).attachmentDirectory, id, BUFFER_SIZE, isLittleEndian()); + if (bytes!=null) { + int nSeries = getSeriesCount(); + ServiceFactory factory = new ServiceFactory(); + OMEXMLService service = factory.getInstance(OMEXMLService.class); + OMEXMLMetadata omeXML = service.createOMEXMLMetadata(); + ZeissCZIReader labelReader = new ZeissCZIReader(); + String placeHolderName = "slide_preview.czi"; + labelReader.setMetadataOptions(getMetadataOptions()); + ByteArrayHandle stream = new ByteArrayHandle(bytes); + Location.mapFile(placeHolderName, stream); + labelReader.setMetadataStore(omeXML); + labelReader.setId(placeHolderName); + + CoreMetadata c = labelReader.getCoreMetadataList().get(0); + + if (c.sizeZ > 1 || c.sizeT > 1) { + return; } - s += core.get(s).resolutionCount; - } - } + core.add(new CoreMetadata(c)); + core.get(core.size() - 1).thumbnail = true; + extraImages.add(labelReader.openBytes(0)); + stream.close(); + coreIndexToSeries.put(core.size() - 1, nSeries); + Location.mapFile(placeHolderName, null); - // check for PALM data; requires planes to split into separate series - - String firstXML = null; - boolean canSkipXML = true; - String currentPath = new Location(currentId).getAbsolutePath(); - boolean isPALM = false; - if (planes.size() <= 2 && getImageCount() <= 2) { - for (Segment segment : segments) { - String path = new Location(segment.filename).getAbsolutePath(); - if (currentPath.equals(path) && segment instanceof Metadata) { - segment.fillInData(); - String xml = ((Metadata) segment).xml; - xml = XMLTools.sanitizeXML(xml); - if (firstXML == null && canSkipXML) { - firstXML = xml; - } - if (canSkipXML && firstXML.equals(xml)) { - isPALM = checkPALM(xml); - } - else if (!firstXML.equals(xml)) { - canSkipXML = false; - } - ((Metadata) segment).clearXML(); - } + /*allPositionsInformation.slidePreviewLocation = new XYZLength(); + allPositionsInformation.slidePreviewPixelSize = new XYZLength(); + allPositionsInformation.slidePreviewPixelSize.pX = omeXML.getPixelsPhysicalSizeX(0); + allPositionsInformation.slidePreviewPixelSize.pY = omeXML.getPixelsPhysicalSizeY(0); + allPositionsInformation.slidePreviewPixelSize.pZ = omeXML.getPixelsPhysicalSizeZ(0); + allPositionsInformation.slidePreviewLocation.pX = omeXML.getPlanePositionX(0,0); + allPositionsInformation.slidePreviewLocation.pY = omeXML.getPlanePositionY(0,0); + allPositionsInformation.slidePreviewLocation.pZ = omeXML.getPlanePositionZ(0,0);*/ + labelReader.close(); } } - if (isPALM) { - LOGGER.debug("Detected PALM data"); - core.get(0).sizeC = 1; - core.get(0).imageCount = core.get(0).sizeZ * core.get(0).sizeT; - - for (int i=0; i= getImageCount()) { - if (core.size() == 1) { - CoreMetadata second = new CoreMetadata(core.get(0)); - core.add(second); - } - p.coreIndex = 1; - p.planeIndex -= (planes.size() / 2); - core.get(1).sizeX = storedX; - core.get(1).sizeY = storedY; - } - else { - core.get(0).sizeX = storedX; - core.get(0).sizeY = storedY; - } - } - if (core.size() == 2) { - // prevent misidentification of PALM data; each plane should be a different size - if (core.get(0).sizeX == core.get(1).sizeX && - core.get(0).sizeY == core.get(1).sizeY) - { - isPALM = false; - core.remove(1); - core.get(0).sizeC = 2; - core.get(0).imageCount *= getSizeC(); - - for (int i=0; i sortedFileParts, Map cziPartToSegments, String id) throws IOException, FormatException { + for (int filePart: sortedFileParts) { + byte[] jpegBytes = LibCZI.getJPGThumbNailBytes(cziPartToSegments.get(filePart).attachmentDirectory, id, BUFFER_SIZE, isLittleEndian()); - readAttachments(); + if (jpegBytes!=null) { // SKIPS THUMBNAIL FOR LEGACY COMPATIBILITY + JPEGReader thumbReader = new JPEGReader(); + String placeHolderName = "image.jpg"; + //thumbReader.setMetadataOptions(getMetadataOptions()); + ByteArrayHandle stream = new ByteArrayHandle(jpegBytes); + Location.mapFile(placeHolderName, stream); + thumbReader.setId(placeHolderName); - // populate the OME metadata + CoreMetadata c = thumbReader.getCoreMetadataList().get(0); - store = makeFilterMetadata(); - MetadataTools.populatePixels(store, this, true); + if (c.sizeZ > 1 || c.sizeT > 1) { - firstXML = null; - canSkipXML = true; - for (Segment segment : segments) { - String path = new Location(segment.filename).getAbsolutePath(); - if (currentPath.equals(path) && segment instanceof Metadata) { - segment.fillInData(); - String xml = ((Metadata) segment).xml; - xml = XMLTools.sanitizeXML(xml); - if (firstXML == null && canSkipXML) { - firstXML = xml; - } - if (canSkipXML && firstXML.equals(xml)) { - translateMetadata(xml); - } - else if (!firstXML.equals(xml)) { - canSkipXML = false; - } - ((Metadata) segment).clearXML(); - } - else if (segment instanceof Attachment) { - AttachmentEntry entry = ((Attachment) segment).attachment; - String name = entry.name.trim(); - - if (name.equals("TimeStamps")) { - segment.fillInData(); - RandomAccessInputStream s = - new RandomAccessInputStream(((Attachment) segment).attachmentData); - try { - s.order(isLittleEndian()); - s.seek(8); - while (s.getFilePointer() + 8 <= s.length()) { - timestamps.add(s.readDouble()); - } - } - finally { - s.close(); + } else { + if ((c.sizeX>1) && (c.sizeY>1)) { // Sometimes there's nothing in the thumbnail + core.add(new CoreMetadata(c)); + core.get(core.size() - 1).thumbnail = true; + extraImages.add(thumbReader.openBytes(0)); + thumbReader.close(); + stream.close(); } } + Location.mapFile(placeHolderName, null); } - segment.close(); } + } - if (rotationLabels != null) { - ms0.moduloZ.labels = rotationLabels; - ms0.moduloZ.end = ms0.moduloZ.start; - } - if (illuminationLabels != null) { - ms0.moduloC.labels = illuminationLabels; - ms0.moduloC.end = ms0.moduloC.start; - } - if (phaseLabels != null) { - ms0.moduloT.labels = phaseLabels; - ms0.moduloT.end = ms0.moduloT.start; + // The parent method does not work well + @Override + public int coreIndexToSeries(int index){ + return coreIndexToSeries.get(index); + } + + /* + This method goes through all subblocks of a particular core index and finds the + X, Y, Z, C, T bounds of these subblocks. + */ + private int[] setOriginAndSize(CoreMetadata ms0, + List blocks, + int minX_maxRes, int minY_maxRes, int nPixX_maxRes, int nPixY_maxRes) { + int minX = Integer.MAX_VALUE; + int minY = Integer.MAX_VALUE; + int minZ = Integer.MAX_VALUE; + int minC = Integer.MAX_VALUE; + int minT = Integer.MAX_VALUE; + int maxX = -Integer.MAX_VALUE; + int maxY = -Integer.MAX_VALUE; + int maxZ = -Integer.MAX_VALUE; + int maxC = -Integer.MAX_VALUE; + int maxT = -Integer.MAX_VALUE; + int downScale = blocks.get(0).getDownSampling(); + + for (ModuloDimensionEntries block: blocks) { + int blockSizeX = block.getDimension("X").storedSize; // size in pixel + int blockSizeY = block.getDimension("Y").storedSize; + if (blockSizeX>maxBlockSizeX) maxBlockSizeX = blockSizeX; + if (blockSizeX>maxBlockSizeY) maxBlockSizeY = blockSizeY; + + int x_min = block.getDimension("X").start/downScale; + int x_max = x_min+blockSizeX; // size or stored size ? + int y_min = block.getDimension("Y").start/downScale; + int y_max = y_min+blockSizeY; + int z_min = 0; + int z_max = 1; + if (block.hasDimension("Z")) { + z_min = block.getDimension("Z").start; + z_max = z_min + block.getDimension("Z").storedSize; + } + int c_min = 0; + int c_max = 1; + if (block.hasDimension("C")) { + c_min = block.getDimension("C").start; + c_max = c_min + block.getDimension("C").storedSize; + } + int t_min = 0; + int t_max = 1; + if (block.hasDimension("T")) { + t_min = block.getDimension("T").start; + t_max = t_min + block.getDimension("T").storedSize; + } + if (maxXx_min) minX = x_min; + if (minY>y_min) minY = y_min; + if (minZ>z_min) minZ = z_min; + if (minC>c_min) minC = c_min; + if (minT>t_min) minT = t_min; + } + + ms0.sizeZ = maxZ - minZ; + ms0.sizeC = maxC - minC; + ms0.sizeT = maxT - minT; + + if ((downScale!=1)&&(allowAutostitching())) { + ms0.sizeX = nPixX_maxRes/downScale; + ms0.sizeY = nPixY_maxRes/downScale; + int[] originCoordinates = new int[2]; + originCoordinates[0] = minX_maxRes/downScale; + originCoordinates[1] = minY_maxRes/downScale; + return originCoordinates; + } else { + ms0.sizeX = maxX - minX; + ms0.sizeY = maxY - minY; + int[] originCoordinates = new int[2]; + originCoordinates[0] = minX; + originCoordinates[1] = minY; + return originCoordinates; } + } - for (int i=0; i indices = new ArrayList(); - if (indexIntoPlanes.containsKey(c)) { - indices = indexIntoPlanes.get(c); - } - indices.add(i); - indexIntoPlanes.put(c, indices); - //Add series metadata : populate position list - int nameWidth = String.valueOf(getSeriesCount()).length(); - for (DimensionEntry dimension : p.directoryEntry.dimensionEntries) { - if (dimension == null) { - continue; - } - switch (dimension.dimension.charAt(0)) { - case 'S': - setCoreIndex(p.coreIndex); - int seriesId = p.coreIndex + 1; - //add padding to make sure the original metadata table is organized properly in ImageJ - String sIndex = String.format("Positions|Series %0" + nameWidth + "d|", seriesId); - if (maxResolution == 0) { - addSeriesMetaList(sIndex, dimension.start); - } - else { - // don't store the start value for every tile in a pyramid - addSeriesMeta(sIndex, dimension.start); - } - break; - } - } - setCoreIndex(0); + private static void convertPixelType(CoreMetadata ms0, int pixelType) throws FormatException { + switch (pixelType) { + case LibCZI.GRAY8: + ms0.pixelType = FormatTools.UINT8; + break; + case LibCZI.GRAY16: + ms0.pixelType = FormatTools.UINT16; + break; + case LibCZI.GRAY32: + ms0.pixelType = FormatTools.UINT32; + break; + case LibCZI.GRAY_FLOAT: + ms0.pixelType = FormatTools.FLOAT; + break; + case LibCZI.GRAY_DOUBLE: + ms0.pixelType = FormatTools.DOUBLE; + break; + case LibCZI.BGR_24: + ms0.pixelType = FormatTools.UINT8; + ms0.sizeC *= 3; + ms0.rgb = true; + ms0.interleaved = true; + break; + case LibCZI.BGR_48: + ms0.pixelType = FormatTools.UINT16; + ms0.sizeC *= 3; + ms0.rgb = true; + ms0.interleaved = true; + break; + case LibCZI.BGRA_8: + ms0.pixelType = FormatTools.UINT8; + ms0.sizeC *= 4; + ms0.rgb = true; + ms0.interleaved = true; + break; + case LibCZI.BGR_FLOAT: + ms0.pixelType = FormatTools.FLOAT; + ms0.sizeC *= 3; + ms0.rgb = true; + ms0.interleaved = true; + break; + case LibCZI.COMPLEX: + case LibCZI.COMPLEX_FLOAT: + throw new FormatException("Sorry, complex pixel data not supported."); + default: + throw new FormatException("Unknown pixel type: " + pixelType); } + ms0.interleaved = ms0.rgb; + } - if (channels.size() > 0 && channels.get(0).color != null && !isRGB()) { - for (int i=0; i 0 && plateColumns > 0 && platePositions.size() > 0) { - store.setPlateID(MetadataTools.createLSID("Plate", 0), 0); - store.setPlateRows(new PositiveInteger(plateRows), 0); - store.setPlateColumns(new PositiveInteger(plateColumns), 0); + /* + * Defines how dimension are sorted for series / core + * This sets in which order the series are created depending on their dimension + * This order has been set to match the original ZeissCZIReader + */ + private static int dimensionPriority(String dimension) { + switch (dimension) { + case "X": + return 0; + case "Y": + return 1; + case "Z": + return 2; + case "T": + return 3; + case RESOLUTION_LEVEL_DIMENSION: + return 4; + case "C": // Channel + return 5; + case "R": // Rotation + return 6; + case "I": // Illumination + return 7; + case "H": // Phase + return 8; + case "V": // View : = Angle + return 9; + case "B": // Block - deprecated + return 10; + case "S": // Scene // That's weird the priority between scene on mosaic, I need to understand a bit better + return 11; + case "M": // Mosaic + return 12; + case FILE_PART_DIMENSION: // File part : number one + return 13; + default: + throw new UnsupportedOperationException("Unknown dimension "+dimension); + } + } - int fieldsPerWell = fieldNames.size() / platePositions.size(); - if (fieldNames.size() == 0) { - fieldsPerWell = 1; - } + // ------------- Extra classes - int nextWell = 0; - int nextField = 0; - for (int i=0, img=0; img this will make all sub-blocks belong to the + * same series. + *

+ * If auto-stitching is true, the mosaic dimension is also ignored, and this will fuse all mosaic blocks + * into a single core series, effectively merging mosaic into a single image. + *

+ * Also, the signature is made with ordering of the dimension, this will allow to sort series according + * to the String signature possible. + *

+ * To avoid issues with ordering, the maximal number of digits per dimension should be known in advance. + */ + static class CoreSignature implements Comparable { + final String signature; + final int hashCode; + + final int filePart; + + public int getFilePart() { + return filePart; + } + public CoreSignature(ModuloDimensionEntries entries, //LibCZI.SubBlockSegment.SubBlockSegmentData.SubBlockDirectoryEntryDV.DimensionEntry[] entries, + String pyramidLevelDimension, int pyramidLevelValue, + Function maxDigitPerDimension, boolean autostitch, + String filePartDimension, int filePartValue) { + this.filePart = filePartValue; + final StringBuilder signatureBuilder = new StringBuilder(); + //signatureBuilder.append(filePartDimension); + //String digitFormat = "%0"+maxDigitPerDimension.apply(filePartDimension)+"d"; + //signatureBuilder.append(String.format(digitFormat, filePartValue)); + entries.getList().stream() + .sorted(Comparator.comparing(e -> dimensionPriority(e.getDimension()))) + .forEachOrdered(e -> { + if (!ignoreDimForSeries(e.getDimension(), autostitch)) { + String digitFormat_inner = "%0"+maxDigitPerDimension.apply(e.getDimension())+"d"; + signatureBuilder.append(e.getDimension()) + .append(String.format(digitFormat_inner, e.getStart())); + } + }); + // TODO : put this as a dimension entry directly + signatureBuilder.append(pyramidLevelDimension); + String digitFormat = "%0"+maxDigitPerDimension.apply(pyramidLevelDimension)+"d"; + signatureBuilder.append(String.format(digitFormat, pyramidLevelValue)); + signature = signatureBuilder.toString(); + hashCode = Objects.hash(signature); // final, so let's precompute the hashcode + } - try { - row = Integer.parseInt(index[0]) - 1; - column = Integer.parseInt(index[1]) - 1; - } - catch (NumberFormatException e) { - LOGGER.trace("Could not parse well position", e); - } + public Map getDimensions() { + Map resultMap = new HashMap<>(); - int field = 0; - if (i < fieldNames.size()) { - String fieldName = fieldNames.get(i); - try { - field = Integer.parseInt(fieldName.substring(1)) - 1; // name starts with "P" - } - catch (NumberFormatException e) { - LOGGER.warn("Could not parse field name {}; plate layout may be incorrect", fieldName); - } - } + // Iterate over the characters in the input string + int startIndex = 0; + int endIndex = startIndex+1; + //for (int i = 0; i < signature.length(); i++) { + while (startIndex= 0 && column >= 0) { - int imageIndex = coreIndexToSeries(img); - store.setWellID(MetadataTools.createLSID("Well", 0, nextWell), 0, nextWell); - store.setWellRow(new NonNegativeInteger(row), 0, nextWell); - store.setWellColumn(new NonNegativeInteger(column), 0, nextWell); - store.setWellSampleID(MetadataTools.createLSID("WellSample", 0, nextWell, nextField), 0, nextWell, nextField); - store.setWellSampleImageRef(MetadataTools.createLSID("Image", imageIndex), 0, nextWell, nextField); - store.setWellSampleIndex(new NonNegativeInteger(imageIndex), 0, nextWell, nextField); - - nextField++; - if (nextField == fieldsPerWell) { - nextField = 0; - nextWell++; - } - } + // Find the index of the digit character after the alphabetical characters + while (endIndex < signature.length() && !Character.isDigit(signature.charAt(endIndex))) { + endIndex++; } - } - } - String experimenterID = MetadataTools.createLSID("Experimenter", 0); - store.setExperimenterID(experimenterID, 0); - store.setExperimenterEmail(userEmail, 0); - store.setExperimenterFirstName(userFirstName, 0); - store.setExperimenterInstitution(userInstitution, 0); - store.setExperimenterLastName(userLastName, 0); - store.setExperimenterMiddleName(userMiddleName, 0); - store.setExperimenterUserName(userName, 0); - - String name = new Location(getCurrentFile()).getName(); - if (imageName != null && imageName.trim().length() > 0) { - name = imageName; - } + // Extract the dimension substring and convert it to an integer value + String dimension = signature.substring(startIndex, endIndex); + startIndex = endIndex; - store.setInstrumentID(MetadataTools.createLSID("Instrument", 0), 0); + while (endIndex < signature.length() && Character.isDigit(signature.charAt(endIndex))) { + endIndex++; + } - int indexLength = String.valueOf(getSeriesCount()).length(); - int positionIndex = -1; - for (int i=0; i= 0) { - continue; } + return resultMap; + } - if (description != null && description.length() > 0) { - store.setImageDescription(description, i); - } + @Override + public boolean equals(Object o) { + if (this == o) return true; + if (o == null || getClass() != o.getClass()) return false; + CoreSignature that = (CoreSignature) o; + return signature.equals(that.signature); + } - if (airPressure != null) { - store.setImagingEnvironmentAirPressure( - new Pressure(new Double(airPressure), UNITS.MILLIBAR), i); - } - if (co2Percent != null) { - store.setImagingEnvironmentCO2Percent( - PercentFraction.valueOf(co2Percent), i); - } - if (humidity != null) { - store.setImagingEnvironmentHumidity( - PercentFraction.valueOf(humidity), i); - } - if (temperature != null) { - store.setImagingEnvironmentTemperature(new Temperature( - new Double(temperature), UNITS.CELSIUS), i); - } + @Override + public int hashCode() { + return hashCode; + } - if (objectiveSettingsID != null) { - store.setObjectiveSettingsID(objectiveSettingsID, i); - if (correctionCollar != null) { - store.setObjectiveSettingsCorrectionCollar( - new Double(correctionCollar), i); - } - if (medium != null) { - store.setObjectiveSettingsMedium(MetadataTools.getMedium(medium), i); - } - if (refractiveIndex != null) { - store.setObjectiveSettingsRefractiveIndex( - new Double(refractiveIndex), i); + @Override + public int compareTo(CoreSignature o) { + return this.signature.compareTo(o.signature); // Sort based on string + } + + @Override + public String toString() { + return signature; + } + + } + + /** + * A class that transforms dimension entries to account for the Modulo implementation + * So it just keeps what's needed and apply modulo operations on C Z and T + */ + static class ModuloDimensionEntries { + /** + // set modulo annotations + // rotations -> modulo Z + // illuminations -> modulo C + // phases -> modulo T + + LOGGER.trace("rotations = {}", rotations); + LOGGER.trace("illuminations = {}", illuminations); + LOGGER.trace("phases = {}", phases); + + LOGGER.trace("positions = {}", positions); + LOGGER.trace("acquisitions = {}", acquisitions); + LOGGER.trace("mosaics = {}", mosaics); + LOGGER.trace("angles = {}", angles); + + ms0.moduloZ.step = ms0.sizeZ; + ms0.moduloZ.end = ms0.sizeZ * (rotations - 1); + ms0.moduloZ.type = FormatTools.ROTATION; + ms0.sizeZ *= rotations; + + ms0.moduloC.step = ms0.sizeC; + ms0.moduloC.end = ms0.sizeC * (illuminations - 1); + ms0.moduloC.type = FormatTools.ILLUMINATION; + ms0.moduloC.parentType = FormatTools.CHANNEL; + ms0.sizeC *= illuminations; + + ms0.moduloT.step = ms0.sizeT; + ms0.moduloT.end = ms0.sizeT * (phases - 1); + ms0.moduloT.type = FormatTools.PHASE; + ms0.sizeT *= phases; + */ + + final List entryList = new ArrayList<>(); + final int nRotations, nIlluminations, nPhases, filePart; + + public ModuloDimensionEntries(LibCZI.SubBlockDirectorySegment.SubBlockDirectorySegmentData.SubBlockDirectoryEntry entry, + int nRotations, int nIlluminations, int nPhases, int nChannels, int nSlices, int nFrames, int filePart) { + this.filePart = filePart; + this.nRotations = nRotations; + this.nIlluminations = nIlluminations; + this.nPhases = nPhases; + this.pixelType = entry.getPixelType(); + this.compression = entry.getCompression(); + this.downSampling = entry.getDimension("X").size/entry.getDimension("X").storedSize; + this.filePosition = entry.getFilePosition(); + + int iRotation = 0; + int iIllumination = 0; + int iPhase = 0; + // Collect + LibCZI.SubBlockSegment.SubBlockSegmentData.SubBlockDirectoryEntryDV.DimensionEntry[] entries = entry.getDimensionEntries(); + for (int i = 0; i index = indexIntoPlanes.get(coordinate); - if (index == null) { - continue; - } + public Collection getList() { + return entryList; + } - SubBlock p = planes.get(index.get(0)); - if (startTime == null) { - startTime = p.timestamp; - } + final int pixelType; - if (firstPlane) { - if (!hasFlattenedResolutions()) { - positionIndex = i; - } - else if (p.resolutionIndex == 0) { - positionIndex++; - } - firstPlane = false; - } + public int getPixelType() { + return pixelType; + } - Double minStageX = null; - Double maxStageX = null; - Double minStageY = null; - Double maxStageY = null; - for (Integer q : index) { - SubBlock currentPlane = planes.get(q); - if (currentPlane == null) continue; - if (storeRelativePositions()) { - if (minStageX == null || currentPlane.col < minStageX) { - minStageX = (double) currentPlane.col; - } - if (maxStageX == null || currentPlane.col > maxStageX) { - maxStageX = (double) currentPlane.col; - } - } - else if (currentPlane.stageX != null) { - if (minStageX == null || - currentPlane.stageX.value().doubleValue() < minStageX) - { - minStageX = currentPlane.stageX.value().doubleValue(); - } - if (maxStageX == null || - currentPlane.stageX.value().doubleValue() > maxStageX) - { - maxStageX = currentPlane.stageX.value().doubleValue(); - } - } - if (storeRelativePositions()) { - if (minStageY == null || currentPlane.row < minStageY) { - minStageY = (double) currentPlane.row; - } - if (maxStageY == null || currentPlane.row > maxStageY) { - maxStageY = (double) currentPlane.row; - } - } - else if (currentPlane.stageY != null) { - if (minStageY == null || - currentPlane.stageY.value().doubleValue() < minStageY) - { - minStageY = currentPlane.stageY.value().doubleValue(); - } - if (maxStageY == null || - currentPlane.stageY.value().doubleValue() > maxStageY) - { - maxStageY = currentPlane.stageY.value().doubleValue(); - } - } - } + final int compression; + public Integer getCompression() { + return compression; + } - // if the XML-defined positions are used, - // assign the same position to each resolution in a pyramid + final int downSampling; + public int getDownSampling() { + return downSampling; + } - Length x = null; - if (storeRelativePositions()) { - x = new Length(minStageX, UNITS.PIXEL); - } - else if (minStageX != null && maxStageX != null) { - double diff = (maxStageX - minStageX) / 2; - x = new Length(minStageX + diff, UNITS.MICROMETER); - if (positionsX != null) { - positionsX[positionIndex] = x; - } - } - else if (positionsX != null && positionIndex < positionsX.length && - positionsX[positionIndex] != null) - { - x = positionsX[positionIndex]; - } - else if (p.stageX != null) { - x = p.stageX; - } - else { - x = new Length(p.col, UNITS.REFERENCEFRAME); - } - if (x != null) { - store.setPlanePositionX(x, i, plane); - if (plane == 0) { - store.setStageLabelX(x, i); - } + public ModuloDimensionEntry getDimension(String dim) { + for (ModuloDimensionEntry entry:getList()) { + if (entry.dimension.equals(dim)) { + return entry; } + } + throw new IllegalArgumentException("No dimension "+dim+" found"); + } - Length y = null; - if (storeRelativePositions()) { - y = new Length(minStageY, UNITS.PIXEL); - } - else if (minStageY != null && maxStageY != null) { - double diff = (maxStageY - minStageY) / 2; - y = new Length(minStageY + diff, UNITS.MICROMETER); - if (positionsY != null) { - positionsY[positionIndex] = y; - } - } - else if (positionsY != null && positionIndex < positionsY.length && - positionsY[positionIndex] != null) - { - y = positionsY[positionIndex]; - } - else if (p.stageY != null) { - y = p.stageY; - } - else { - y = new Length(p.row, UNITS.REFERENCEFRAME); - } - if (y != null) { - store.setPlanePositionY(y, i, plane); - if (plane == 0) { - store.setStageLabelY(y, i); - } - } + final long filePosition; + public long getFilePosition() { + return filePosition; + } - Length z = null; - if (p.stageZ != null) { - z = p.stageZ; - } - else if (positionsZ != null && positionIndex < positionsZ.length) { - int zIndex = getZCTCoords(plane)[0]; - if (positionsZ[positionIndex] != null) { - if (zStep != null) { - double value = positionsZ[positionIndex].value(zStep.unit()).doubleValue(); - if (zStep != null) { - value += zIndex * zStep.value().doubleValue(); - } - z = new Length(value, zStep.unit()); - } - else { - z = positionsZ[positionIndex]; - } - } - } - if (z != null) { - store.setPlanePositionZ(z, i, plane); - if (plane == 0) { - store.setStageLabelZ(z, i); - } - } - if (plane == 0 && (x != null || y != null || z != null)) { - store.setStageLabelName("Scene position #" + i, i); + public boolean hasDimension(String dim) { + for (ModuloDimensionEntry entry:getList()) { + if (entry.dimension.equals(dim)) { + return true; } + } + return false; + } - if (p.timestamp != null) { - store.setPlaneDeltaT(new Time(p.timestamp - startTime, UNITS.SECOND), i, plane); - } - else if (plane < timestamps.size() && timestamps.size() == getImageCount()) { - // only use the plane index if there is one timestamp per plane - if (timestamps.get(plane) != null) { - store.setPlaneDeltaT(new Time(timestamps.get(plane), UNITS.SECOND), i, plane); - } - } - else if (getZCTCoords(plane)[2] < timestamps.size()) { - // otherwise use the timepoint index, to prevent incorrect timestamping of channels - int t = getZCTCoords(plane)[2]; - if (timestamps.get(t) != null) { - store.setPlaneDeltaT(new Time(timestamps.get(t), UNITS.SECOND), i, plane); - } - } - if (p.exposureTime != null) { - store.setPlaneExposureTime(new Time(p.exposureTime, UNITS.SECOND), i, plane); - } - else { - int channel = getZCTCoords(plane)[1]; - if (channel < channels.size() && - channels.get(channel).exposure != null) - { - store.setPlaneExposureTime( - new Time(channels.get(channel).exposure, UNITS.SECOND), i, plane); - } - } + static class ModuloDimensionEntry { + //LibCZI.SubBlockSegment.SubBlockSegmentData.SubBlockDirectoryEntryDV.DimensionEntry entry; + final String dimension; + final int start; + final int storedSize; + public ModuloDimensionEntry(String dimension, int start, int storedSize) { + this.dimension = dimension; + this.start = start; + this.storedSize = storedSize; } - if (firstPlane && core.get(i).resolutionCount > 1 && - hasFlattenedResolutions()) - { - positionIndex++; + public String getDimension() { + return dimension; } - for (int c=0; c 6) { - color = color.substring(2, color.length()); - } - try { - // shift by 8 to allow alpha in the final byte - store.setChannelColor( - new Color((Integer.parseInt(color, 16) << 8) | 0xff), i, c); - } - catch (NumberFormatException e) { - LOGGER.warn("", e); - } - } + /** + * This is a class that wraps three numbers c,z,t and an object + * can be used as a key in a hashmap. + *

+ * It's used to create a Map from CZTKey to Blocks instead of + * Map from C to Map from Z to Map from T to Blocks + */ + static class CZTKey { + public final int c,z,t; + public final int hashCode; + public CZTKey(int c, int z, int t) { + this.c = c; + this.z = z; + this.t = t; + hashCode = Objects.hash(c,z,t); + } - String emWave = channels.get(c).emission; - if (emWave != null) { - Double wave = new Double(emWave); - Length em = FormatTools.getEmissionWavelength(wave); - if (em != null) { - store.setChannelEmissionWavelength(em, i, c); - } - } - String exWave = channels.get(c).excitation; - if (exWave != null) { - Double wave = new Double(exWave); - Length ex = FormatTools.getExcitationWavelength(wave); - if (ex != null) { - store.setChannelExcitationWavelength(ex, i, c); - } - } + @Override + public boolean equals(Object o) { + if (this == o) return true; + if (o == null || getClass() != o.getClass()) return false; + CZTKey that = (CZTKey) o; + return (that.z == this.z)&&(that.c == this.c)&&(that.t == this.t); + } + @Override + public int hashCode() { + return hashCode; + } - if (channels.get(c).illumination != null) { - store.setChannelIlluminationType( - channels.get(c).illumination, i, c); - } + @Override + public String toString() { + return "C:"+c+"; Z:"+z+"; T:"+t; + } + } - if (channels.get(c).pinhole != null) { - store.setChannelPinholeSize( - new Length(new Double(channels.get(c).pinhole), UNITS.MICROMETER), i, c); - } - if (channels.get(c).acquisitionMode != null) { - store.setChannelAcquisitionMode( - channels.get(c).acquisitionMode, i, c); - } - } - if (c < detectorRefs.size()) { - String detector = detectorRefs.get(c); - store.setDetectorSettingsID(detector, i, c); + /** + * A stripped down version of + * {@link LibCZI.SubBlockDirectorySegment.SubBlockDirectorySegmentData.SubBlockDirectoryEntry} + * Because we have really many of these objects, and it's critical to keep these objects as small as possible + */ + static class MinDimEntry { - if (c < binnings.size()) { - store.setDetectorSettingsBinning(MetadataTools.getBinning(binnings.get(c)), i, c); - } - if (c < channels.size()) { - store.setDetectorSettingsGain(channels.get(c).gain, i, c); - } - } + final int dimensionStartZ; - if (c < channels.size()) { - if (hasDetectorSettings) { - store.setDetectorSettingsGain(channels.get(c).gain, i, c); - } - } + public MinDimEntry(ModuloDimensionEntries entry) { + filePosition = entry.getFilePosition(); + dimensionStartX = entry.getDimension("X").start; + dimensionStartY = entry.getDimension("Y").start; + if (entry.hasDimension("Z")) { + dimensionStartZ = entry.getDimension("Z").start; + } else { + dimensionStartZ = 0; } + storedSizeX = entry.getDimension("X").storedSize; + storedSizeY = entry.getDimension("Y").storedSize; + filePart = entry.filePart; } - // not needed by further calls on the reader - segments = null; + final long filePosition; + final int dimensionStartX, dimensionStartY; + final int storedSizeX, storedSizeY; + + final int filePart; + + } - // -- ZeissCZI-specific methods -- - public boolean allowAutostitching() { - MetadataOptions options = getMetadataOptions(); - if (options instanceof DynamicMetadataOptions) { - return ((DynamicMetadataOptions) options).getBoolean( - ALLOW_AUTOSTITCHING_KEY, ALLOW_AUTOSTITCHING_DEFAULT); - } - return ALLOW_AUTOSTITCHING_DEFAULT; + //MinimalDimensionEntry - for Java > 17 + /*public record MinDimEntry(int dimensionStartZ, long filePosition, + + int dimensionStartX,int dimensionStartY, int storedSizeX, int storedSizeY, int filePart) {} + + public static MinDimEntry getMinimalEntry(ModuloDimensionEntries entry) { + long filePosition = entry.getFilePosition(); + int dimensionStartX = entry.getDimension("X").start; + int dimensionStartY = entry.getDimension("Y").start; + int dimensionStartZ = 0; + if (entry.hasDimension("Z")) { + dimensionStartZ = entry.getDimension("Z").start; + } + int storedSizeX = entry.getDimension("X").storedSize; + int storedSizeY = entry.getDimension("Y").storedSize; + int filePart = entry.filePart; + return new MinDimEntry(dimensionStartZ, filePosition, dimensionStartX, dimensionStartY, storedSizeX, storedSizeY, filePart); + }*/ + + /** Duplicates this reader for parallel reading. + * Creating a reader with this method allows to keep a very low memory footprint + * because all immutable objects are re-used by reference. + * WARNING: calling {@link ZeissCZIReader#close()} on one of these readers will prevent the use + * of all the other readers created with this method */ + public ZeissCZIReader copy() { + return new ZeissCZIReader(this); } - public boolean canReadAttachments() { - MetadataOptions options = getMetadataOptions(); - if (options instanceof DynamicMetadataOptions) { - return ((DynamicMetadataOptions) options).getBoolean( - INCLUDE_ATTACHMENTS_KEY, INCLUDE_ATTACHMENTS_DEFAULT); - } - return INCLUDE_ATTACHMENTS_DEFAULT; + // An annotation that's helpful to re-initialize all fields in the new reader copied + // from a model one (with the copy method or with the constructor with the reader + // in argument) + @Retention(RetentionPolicy.RUNTIME) + @interface CopyByRef { + } - public boolean trimDimensions() { - MetadataOptions options = getMetadataOptions(); - if (options instanceof DynamicMetadataOptions) { - return ((DynamicMetadataOptions) options).getBoolean( - TRIM_DIMENSIONS_KEY, TRIM_DIMENSIONS_DEFAULT); + /** + * A structure that helps to group all CZI segments + * that are used in the file initialization + */ + private static class CZISegments { + final LibCZI.FileHeaderSegment fileHeader; + final LibCZI.SubBlockDirectorySegment subBlockDirectory; + final LibCZI.AttachmentDirectorySegment attachmentDirectory; + final LibCZI.MetaDataSegment metadata; + final double[] timeStamps; + final String fileName; + public CZISegments(String id, boolean littleEndian) throws IOException { + this.fileName = id; + this.fileHeader = LibCZI.getFileHeaderSegment(id, BUFFER_SIZE, littleEndian); + this.subBlockDirectory = LibCZI.getSubBlockDirectorySegment(this.fileHeader, id, BUFFER_SIZE, littleEndian); + this.metadata = LibCZI.getMetaDataSegment(this.fileHeader, id, BUFFER_SIZE, littleEndian); + this.attachmentDirectory = LibCZI.getAttachmentDirectorySegment(this.fileHeader, id, BUFFER_SIZE, littleEndian); + if (attachmentDirectory!=null) { + this.timeStamps = LibCZI.getTimeStamps(this.attachmentDirectory, id, BUFFER_SIZE, littleEndian); + //System.out.println("#ts="+timeStamps.length); + /*for (double timeStamp: timeStamps) { + System.out.println(timeStamp); + }*/ + } else { + this.timeStamps = new double[0]; + } } - return TRIM_DIMENSIONS_DEFAULT; } - public boolean storeRelativePositions() { - MetadataOptions options = getMetadataOptions(); - if (options instanceof DynamicMetadataOptions) { - return ((DynamicMetadataOptions) options).getBoolean( - RELATIVE_POSITIONS_KEY, RELATIVE_POSITIONS_DEFAULT); + static class SubBlockLRUCache extends + LinkedHashMap> + { + + private static final long serialVersionUID = 1L; + + private long maxCost; + + AtomicLong totalWeight = new AtomicLong(); + + HashMap cost = new HashMap<>(); + + public SubBlockLRUCache(final int iniSize, final long maxCost) { + super(iniSize, 0.75f, true); + this.maxCost = maxCost; } - return RELATIVE_POSITIONS_DEFAULT; - } - // -- Helper methods -- + public void setMaxCost(long maxCost) { + this.maxCost = maxCost; + } - private void readSegments(String id) throws IOException { - if (in != null) { - in.close(); + public long getCost() { + return totalWeight.get(); } - in = new RandomAccessInputStream(id, BUFFER_SIZE); - in.order(isLittleEndian()); - while (in.getFilePointer() < in.length()) { - Segment segment = readSegment(id); - if (segment == null) { - break; + + public long getMaxCost() { + return maxCost; + } + + @Override + protected boolean removeEldestEntry( + final Map.Entry> eldest) + { + if (totalWeight.get() > maxCost) { + totalWeight.addAndGet(-cost.get(eldest.getKey())); + cost.remove(eldest.getKey()); + eldest.getValue().clear(); + //System.out.println("Remove"); + return true; + } + else return false; + } + + synchronized public void touch(final MinDimEntry key, + final byte[] value) + { + final SoftReference ref = get(key); + if (ref == null) { + long costValue = value.length;//getWeight(value); + totalWeight.addAndGet(costValue); + cost.put(key, costValue); + //System.out.println(totalWeight.get()/(1024*1024)+" Mb"); + put(key, new SoftReference<>(value)); + } + else if (ref.get() == null) { + put(key, new SoftReference<>(value)); } - segments.add(segment); + } - if (segment instanceof SubBlock) { - planes.add((SubBlock) segment); - LOGGER.trace("plane #{} = {}", planes.size() - 1, segment); + @Override + public synchronized void clear() { + for (final SoftReference ref : values()) { + ref.clear(); } - segment.close(); + totalWeight.set(0); + cost.clear(); + super.clear(); } } - private void readAttachments() throws FormatException, IOException { - if (!canReadAttachments()) { - return; + /** + * This Zeiss Reader class was initially huge and contained many fields. One issue + * is that these fields were only necessary during the reader initialisation (method initFile) + * and were not needed after. + *

+ * As a consequence, many of these fields were transient: they are not serialized. That was the original + * design. However, this was creating a bit of confusion regarding the accessible and initialized fields that + * you could access after the reader was initialized. + *

+ * This creates a bit of confusion regarding the fields which are necessary during the initialisation only + * and the fields which are necessary just when retrieving data. + *

+ * So, in order to solve this issue and clarify a bit the structure of the reader, all these transient fields + * required for initialisation and the methods containing the logic of the metadata reading / translating + * have been moved into this inner Initializer class. + *

+ * There should be a unique initializer object created in the initFile method, and its scope SHOULD NOT extend beyond + * the initFile method. + *

+ * This could also be a non-static class that do not need to pass the object in the constructor. + */ + private static class MetadataInitializer { + + MetadataInitializer(ZeissCZIReader reader) { + this.reader = reader; } - boolean foundLabel = false; - boolean foundPreview = false; - for (Segment segment : segments) { - if (segment instanceof Attachment) { - AttachmentEntry entry = ((Attachment) segment).attachment; - String name = entry.name.trim(); - - if ((name.equals("Label") && !foundLabel) || - (name.equals("SlidePreview") && !foundPreview)) - { - if (!foundLabel) { - foundLabel = name.equals("Label"); - } - if (!foundPreview) { - foundPreview = name.equals("SlidePreview"); - } - segment.fillInData(); - // label and preview are CZI files embedded as attachments + void initializeMetadata(Map cziPartToSegments, + List< // CoreIndex + HashMap>> + List>> + mapCoreTZCToBlocks) throws FormatException, IOException { + // MetaData from xml file + DocumentBuilder parser = XMLTools.createBuilder(); - ZeissCZIReader thumbReader = new ZeissCZIReader(); - thumbReader.setMetadataOptions(getMetadataOptions()); - ByteArrayHandle stream = new ByteArrayHandle(((Attachment) segment).attachmentData); - Location.mapFile("image.czi", stream); - thumbReader.setId("image.czi"); + // But is everything in the master file ??? TODO: verify whether all xml metadata are in the master file + readXMLMetadata(cziPartToSegments.get(0).metadata, parser); - CoreMetadata c = thumbReader.getCoreMetadataList().get(0); + // Timestamps are already read when creating the CZISegments objects + MetadataTools.populatePixels(reader.store, reader, true); - if (c.sizeZ > 1 || c.sizeT > 1) { - continue; - } + // Needs to set the instrument reference before calling the next method, assumed only one per czi file (or fileset for multi-series files) + reader.store.setInstrumentID(MetadataTools.createLSID("Instrument", 0), 0); - core.add(new CoreMetadata(c)); - core.get(core.size() - 1).thumbnail = true; - ((Attachment) segment).attachmentData = thumbReader.openBytes(0); - thumbReader.close(); + setExperimenterInformation(); - stream.close(); - Location.mapFile("image.czi", null); - extraImages.add((Attachment) segment); - } - } - segment.close(); - } - } + // TODO: is everything in the master file ? + setSpaceAndTimeInformation(mapCoreTZCToBlocks, parser, cziPartToSegments.get(0)); - private void calculateDimensions() { - calculateDimensions(0, false); - } + setImageNames(); - private void calculateDimensions(int coreIndex, boolean xyOnly) { - // calculate the dimensions - CoreMetadata ms0 = core.get(coreIndex); - int previousCoreIndex = getCoreIndex(); - setCoreIndex(coreIndex); + setAdditionalImageMetadata(); - ArrayList uniqueT = new ArrayList(); + setChannelMetadata(); - int minPositions = Integer.MAX_VALUE; - int maxPositions = Integer.MIN_VALUE; + //setExtraImagesSpatialInformation(); - int minX = Integer.MAX_VALUE; - int maxX = Integer.MIN_VALUE; - int minY = Integer.MAX_VALUE; - int maxY = Integer.MIN_VALUE; + setPlateInformation(); - int dimensionCount = 0; - for (SubBlock plane : planes) { - if (xyOnly && plane.coreIndex != coreIndex) { - continue; - } - boolean moreDimensions = plane.directoryEntry.dimensionEntries.length > dimensionCount; - if (moreDimensions) { - dimensionCount = plane.directoryEntry.dimensionEntries.length; - ms0.sizeX = 0; - ms0.sizeY = 0; - } - for (DimensionEntry dimension : plane.directoryEntry.dimensionEntries) { - if (dimension == null) { - continue; - } - if (xyOnly && dimension.dimension.charAt(0) != 'X' && - dimension.dimension.charAt(0) != 'Y') - { - continue; - } - switch (dimension.dimension.charAt(0)) { - case 'X': - plane.x = dimension.size; - plane.col = dimension.start; - minX = (int) Math.min(plane.col, minX); - maxX = (int) Math.max(plane.col + plane.x, maxX); - if ((prestitched == null || prestitched) && - getSizeX() > 0 && dimension.size != getSizeX() && allowAutostitching()) - { - prestitched = true; - continue; - } - if (allowAutostitching() || ms0.sizeX == 0 || dimension.size == dimension.storedSize) { - ms0.sizeX = dimension.size; - } - break; - case 'Y': - plane.y = dimension.size; - plane.row = dimension.start; - minY = (int) Math.min(plane.row, minY); - maxY = (int) Math.max(plane.row + plane.y, maxY); - if ((prestitched == null || prestitched) && - getSizeY() > 0 && dimension.size != getSizeY() && allowAutostitching()) - { - prestitched = true; - continue; - } - if (allowAutostitching() || ms0.sizeY == 0 || dimension.size == dimension.storedSize) { - ms0.sizeY = dimension.size; - } - break; - case 'C': - if (dimension.start >= getSizeC()) { - ms0.sizeC = dimension.start + 1; - } - break; - case 'Z': - if (dimension.start > 0 && dimension.start >= getSizeZ()) { - ms0.sizeZ = dimension.start + 1; - } - else if (dimension.size > getSizeZ()) { - ms0.sizeZ = dimension.size; - } - break; - case 'T': - int startIndex = dimension.start; - int endIndex = startIndex + dimension.size; - - for (int i=startIndex; i= rotations) { - rotations = dimension.start + 1; - } - break; - case 'S': - if (dimension.start < minPositions) { - minPositions = dimension.start; - } - if (dimension.start > maxPositions) { - maxPositions = dimension.start; - } - break; - case 'I': - if (dimension.start >= illuminations) { - illuminations = dimension.start + 1; - } - break; - case 'B': - if (dimension.start >= acquisitions) { - acquisitions = dimension.start + 1; - } - break; - case 'M': - if (dimension.start >= mosaics) { - mosaics = dimension.start + 1; - } - break; - case 'H': - if (dimension.start >= phases) { - phases = dimension.start + 1; - } - break; - case 'V': - if (dimension.start >= angles) { - angles = dimension.start + 1; - } - break; - default: - LOGGER.warn("Unknown dimension '{}'", dimension.dimension); - } - } - } - if (maxPositions > Integer.MIN_VALUE && minPositions < Integer.MAX_VALUE) { - positions = maxPositions - minPositions + 1; + setModuloLabels(reader.core.get(0)); } - if (xyOnly && trimDimensions()) { - ms0.sizeX = maxX - minX; - ms0.sizeY = maxY - minY; - } + final ZeissCZIReader reader; - setCoreIndex(previousCoreIndex); - } + private String userName, + userFirstName, + userLastName, + userMiddleName, + userEmail, + userInstitution; - private void assignPlaneIndices() { - LOGGER.trace("assignPlaneIndices:"); - // assign plane and series indices to each SubBlock + private String temperature, airPressure, humidity, co2Percent; - if (core.size() == mosaics && maxResolution == 0) { - LOGGER.trace(" reset position, acquisition, and angle count"); - positions = 1; - acquisitions = 1; - angles = 1; - } + private String gain; - // use the natural ordering of the extra dimensions, - // instead of always using SBMV - ArrayList extraDimOrder = new ArrayList(); - int[] extraLengths = new int[4]; + private String imageName; - int prevS = 0, prevB = 0, prevM = 0, prevV = 0; - for (int p=0; p prevS) { - if (!extraDimOrder.contains('S')) { - extraLengths[extraDimOrder.size()] = positions; - extraDimOrder.add('S'); - } - } - prevS = dimension.start; - break; - case 'B': - if (dimension.start > prevB) { - if (!extraDimOrder.contains('B')) { - extraLengths[extraDimOrder.size()] = acquisitions; - extraDimOrder.add('B'); - } - } - prevB = dimension.start; - break; - case 'M': - if (dimension.start > prevM) { - if (!extraDimOrder.contains('M') && mosaics <= getSeriesCount() && - (prestitched == null || !prestitched || !allowAutostitching())) - { - extraLengths[extraDimOrder.size()] = mosaics; - extraDimOrder.add('M'); - } - } - prevM = dimension.start; - break; - case 'V': - if (dimension.start > prevV) { - if (!extraDimOrder.contains('V')) { - extraLengths[extraDimOrder.size()] = angles; - extraDimOrder.add('V'); - } - } - prevV = dimension.start; - break; - } - } - } - int allLengths = 1; - for (int len : extraLengths) { - if (len > 0) { - allLengths *= len; + private String acquiredDate; + + private String description; + + private String userDisplayName; + + private String correctionCollar, medium, refractiveIndex; + + private Time timeIncrement; + + final private ArrayList gains = new ArrayList<>(); + + private Length zStep; + + private String objectiveSettingsID; + + private int plateRows; + + private int plateColumns; + + final private ArrayList platePositions = new ArrayList<>(); + + final private ArrayList fieldNames = new ArrayList<>(); + + final private ArrayList imageNames = new ArrayList<>(); + + private Timestamp[] coreIndexTimeStamp; + + final private Map coreToPixSizeX = new HashMap<>(); + final private Map coreToPixSizeY = new HashMap<>(); + final private Map coreToPixSizeZ = new HashMap<>(); // Because I can't read from the store.... RAAHAAH + + final private ArrayList channels = new ArrayList<>(); + + final private ArrayList binnings = new ArrayList<>(); + + private String zoom; + + final private ArrayList detectorRefs = new ArrayList<>(); + + private boolean hasDetectorSettings = false; + + private String[] rotationLabels, phaseLabels, illuminationLabels; + + final AllPositionsInformation allPositionsInformation = new AllPositionsInformation(); + + private void setExperimenterInformation() { + // User information fields + String experimenterID = MetadataTools.createLSID("Experimenter", 0); + reader.store.setExperimenterID(experimenterID, 0); + reader.store.setExperimenterEmail(userEmail, 0); + reader.store.setExperimenterFirstName(userFirstName, 0); + reader.store.setExperimenterInstitution(userInstitution, 0); + reader.store.setExperimenterLastName(userLastName, 0); + reader.store.setExperimenterMiddleName(userMiddleName, 0); + reader.store.setExperimenterUserName(userName, 0); + for (int iSeries=0; iSeries 0 && plateColumns > 0 && platePositions.size() > 0) { + reader.store.setPlateID(MetadataTools.createLSID("Plate", 0), 0); + reader.store.setPlateRows(new PositiveInteger(plateRows), 0); + reader.store.setPlateColumns(new PositiveInteger(plateColumns), 0); + + int fieldsPerWell = fieldNames.size() / platePositions.size(); + if (fieldNames.size() == 0) { + fieldsPerWell = 1; } - int extraIndex = extraDimOrder.indexOf(dimension.dimension.charAt(0)); - switch (dimension.dimension.charAt(0)) { - case 'C': - c = dimension.start - plane.pixelTypeIndex; - break; - case 'Z': - z = dimension.start; - if (z >= getSizeZ()) { - z = getSizeZ() - 1; + + int nextWell = 0; + int nextField = 0; + for (int i=0, img=0; img= getSizeT()) { - t = getSizeT() - 1; + int row = -1; + int column = -1; + + try { + row = Integer.parseInt(index[0]) - 1; + column = Integer.parseInt(index[1]) - 1; } - break; - case 'R': - r = dimension.start; - break; - case 'S': - if (extraIndex >= 0) { - extra[extraIndex] = dimension.start; - if (extra[extraIndex] >= extraLengths[extraIndex]) { - extra[extraIndex] = 0; - } + catch (NumberFormatException e) { + LOGGER.trace("Could not parse well position", e); } - break; - case 'I': - i = dimension.start; - break; - case 'B': - if (extraIndex >= 0) { - extra[extraIndex] = dimension.start; - if (extra[extraIndex] >= extraLengths[extraIndex]) { - extra[extraIndex] = 0; + + int field = 0; + if (i < fieldNames.size()) { + String fieldName = fieldNames.get(i); + try { + field = Integer.parseInt(fieldName.substring(1)) - 1; // name starts with "P" } - } - break; - case 'M': - if (extraIndex >= 0) { - extra[extraIndex] = dimension.start; - if (extra[extraIndex] >= extraLengths[extraIndex]) { - extra[extraIndex] = 0; + catch (NumberFormatException e) { + LOGGER.warn("Could not parse field name {}; plate layout may be incorrect", fieldName); } } - break; - case 'H': - phase = dimension.start; - break; - case 'V': - if (extraIndex >= 0) { - extra[extraIndex] = dimension.start; - if (extra[extraIndex] >= extraLengths[extraIndex]) { - extra[extraIndex] = 0; + + if (row >= 0 && column >= 0) { + int imageIndex = reader.coreIndexToSeries(img); + reader.store.setWellID(MetadataTools.createLSID("Well", 0, nextWell), 0, nextWell); + reader.store.setWellRow(new NonNegativeInteger(row), 0, nextWell); + reader.store.setWellColumn(new NonNegativeInteger(column), 0, nextWell); + reader.store.setWellSampleID(MetadataTools.createLSID("WellSample", 0, nextWell, nextField), 0, nextWell, nextField); + reader.store.setWellSampleImageRef(MetadataTools.createLSID("Image", imageIndex), 0, nextWell, nextField); + reader.store.setWellSampleIndex(new NonNegativeInteger(imageIndex), 0, nextWell, nextField); + + nextField++; + if (nextField == fieldsPerWell) { + nextField = 0; + nextWell++; } } - noAngle = false; - break; + } } } + } - if (angles > 1 && noAngle) { - extra[extraDimOrder.indexOf('V')] = - p / (getImageCount() * (getSeriesCount() / angles)); - } + private void readXMLMetadata(LibCZI.MetaDataSegment metaDataSegment, DocumentBuilder parser) throws FormatException, IOException { + String xml = metaDataSegment.data.xml; + xml = XMLTools.sanitizeXML(xml); + //System.out.println(xml); + translateMetadata(xml, parser); + } - if (rotations > 0) { - z = r * (getSizeZ() / rotations) + z; + private void translateMetadata(String xml, DocumentBuilder parser) throws FormatException, IOException { + Element root; + try { + ByteArrayInputStream s = + new ByteArrayInputStream(xml.getBytes(Constants.ENCODING)); + root = parser.parse(s).getDocumentElement(); + s.close(); } - if (illuminations > 0) { - c = i * (getSizeC() / illuminations) + c; + catch (SAXException e) { + throw new FormatException(e); } - if (phases > 0) { - t = phase * (getSizeT() / phases) + t; + + if (root == null) { + throw new FormatException("Could not parse the XML metadata."); } - plane.planeIndex = getIndex(z, c, t); - if (plane.pixelTypeIndex > 0) { - plane.coreIndex = plane.pixelTypeIndex; + NodeList children = root.getChildNodes(); + Element realRoot = null; + for (int i=0; i nameStack = new ArrayDeque<>(); + populateOriginalMetadata(realRoot, nameStack); } - NodeList children = root.getChildNodes(); - Element realRoot = null; - for (int i=0; i nameStack = new ArrayDeque(); - populateOriginalMetadata(realRoot, nameStack); - } + // ---------------- POSITIONS + readPositions(acquisition); - private boolean checkPALM(String xml) throws FormatException, IOException { - Element root = null; - try { - ByteArrayInputStream s = - new ByteArrayInputStream(xml.getBytes(Constants.ENCODING)); - root = parser.parse(s).getDocumentElement(); - s.close(); - } - catch (SAXException e) { - throw new FormatException(e); - } + // ---------------- DETECTORS + { + NodeList detectors = getGrandchildren(acquisition, "Detector"); + + Element setup = getFirstNode(acquisition, "AcquisitionModeSetup"); + String cameraModel = getFirstNodeValue(setup, "SelectedCamera"); + + if (detectors != null) { + for (int i = 0; i < detectors.getLength(); i++) { + Element detector = (Element) detectors.item(i); + String id = MetadataTools.createLSID("Detector", 0, i); + + reader.store.setDetectorID(id, 0, i); + String model = detector.getAttribute("Id"); + reader.store.setDetectorModel(model, 0, i); + + String bin = getFirstNodeValue(detector, "Binning"); + if (bin != null) { + bin = bin.replaceAll(",", "x"); + Binning binning = MetadataTools.getBinning(bin); + + if (model.equals(cameraModel)) { + for (int image = 0; image < reader.getSeriesCount(); image++) { + for (int c = 0; c < reader.getEffectiveSizeC(); c++) { + reader.store.setDetectorSettingsID(id, image, c); + reader.store.setDetectorSettingsBinning(binning, image, c); + } + } + hasDetectorSettings = true; + } + } + } + } - if (root == null) { - throw new FormatException("Could not parse the XML metadata."); - } + Element multiTrack = getFirstNode(acquisition, "MultiTrackSetup"); + + if (multiTrack == null) { + return; + } - NodeList customAttributes = root.getElementsByTagName("CustomAttributes"); - if (customAttributes != null && customAttributes.getLength() > 0) { - Element attributes = (Element) customAttributes.item(0); - if (attributes != null) { - NodeList lsmTags = attributes.getElementsByTagName("LsmTag"); - if (lsmTags != null) { - for (int i=0; i 0) { + for (int i = 0; i < detectors.getLength(); i++) { + Element detector = (Element) detectors.item(i); + String voltage = getFirstNodeValue(detector, "Voltage"); + if (i == 0 && d == 0) { + gain = voltage; + } + gains.add(voltage); } } } - } - } - - NodeList experiments = root.getElementsByTagName("Experiment"); - if (experiments == null || experiments.getLength() == 0) { - return false; - } - Element experimentBlock = - getFirstNode((Element) experiments.item(0), "ExperimentBlocks"); - Element acquisition = getFirstNode(experimentBlock, "AcquisitionBlock"); - if (acquisition == null) { - return false; - } + NodeList tracks = multiTrack.getElementsByTagName("Track"); - Element multiTrack = getFirstNode(acquisition, "MultiTrackSetup"); - if (multiTrack == null) { - return false; - } - Element trackSetup = getFirstNode(multiTrack, "TrackSetup"); - if (trackSetup == null) { - return false; - } - Element palmSlider = getFirstNode(trackSetup, "PalmSlider"); - if (palmSlider == null) { - return false; - } - return Boolean.parseBoolean(palmSlider.getTextContent()); - } + if (tracks.getLength() > 0) { + for (int i = 0; i < tracks.getLength(); i++) { + Element track = (Element) tracks.item(i); + Element channel = getFirstNode(track, "Channel"); + String exposure = getFirstNodeValue(channel, "ExposureTime"); + String gain = getFirstNodeValue(channel, "EMGain"); - private void translateInformation(Element root) throws FormatException { - NodeList informations = root.getElementsByTagName("Information"); - if (informations == null || informations.getLength() == 0) { - return; - } + while (channels.size() <= i) { + channels.add(new Channel()); + } - Element information = (Element) informations.item(0); - Element image = getFirstNode(information, "Image"); - Element user = getFirstNode(information, "User"); - Element environment = getFirstNode(information, "Environment"); - Element instrument = getFirstNode(information, "Instrument"); - Element document = getFirstNode(information, "Document"); - - if (image != null) { - String bitCount = getFirstNodeValue(image, "ComponentBitCount"); - if (bitCount != null) { - core.get(0).bitsPerPixel = Integer.parseInt(bitCount); + try { + if (exposure != null) { + channels.get(i).exposure = Double.valueOf(exposure); + } + } catch (NumberFormatException e) { + LOGGER.debug("Could not parse exposure time", e); + } + try { + if (gain != null) { + channels.get(i).gain = Double.valueOf(gain); + } + } catch (NumberFormatException e) { + LOGGER.debug("Could not parse gain", e); + } + } + } } + } - acquiredDate = getFirstNodeValue(image, "AcquisitionDateAndTime"); - - Element objectiveSettings = getFirstNode(image, "ObjectiveSettings"); - correctionCollar = - getFirstNodeValue(objectiveSettings, "CorrectionCollar"); - medium = getFirstNodeValue(objectiveSettings, "Medium"); - refractiveIndex = getFirstNodeValue(objectiveSettings, "RefractiveIndex"); + private void translateInformation(Element root) throws FormatException { + MetadataStore store = reader.store; - String sizeV = getFirstNodeValue(image, "SizeV"); - if (sizeV != null && angles == 1) { - angles = Integer.parseInt(sizeV); + NodeList informations = root.getElementsByTagName("Information"); + if (informations.getLength() == 0) { + return; } - Element dimensions = getFirstNode(image, "Dimensions"); - - Element tNode = getFirstNode(dimensions, "T"); - if (tNode != null) { - Element positions = getFirstNode(tNode, "Positions"); - if (positions != null) { - Element interval = getFirstNode(positions, "Interval"); - if (interval != null) { - Element incrementNode = getFirstNode(interval, "Increment"); - if (incrementNode != null) { - String increment = incrementNode.getTextContent(); - timeIncrement = new Time(DataTools.parseDouble(increment), UNITS.SECOND); + Element information = (Element) informations.item(0); + Element image = getFirstNode(information, "Image"); + Element user = getFirstNode(information, "User"); + Element environment = getFirstNode(information, "Environment"); + Element instrument = getFirstNode(information, "Instrument"); + Element document = getFirstNode(information, "Document"); + + if (image != null) { + String bitCount = getFirstNodeValue(image, "ComponentBitCount"); + if (bitCount != null) { + reader.core.get(0).bitsPerPixel = Integer.parseInt(bitCount); + } // TODO: understand if the line above is necessary + + acquiredDate = getFirstNodeValue(image, "AcquisitionDateAndTime"); + + Element objectiveSettings = getFirstNode(image, "ObjectiveSettings"); + correctionCollar = + getFirstNodeValue(objectiveSettings, "CorrectionCollar"); + medium = getFirstNodeValue(objectiveSettings, "Medium"); + refractiveIndex = getFirstNodeValue(objectiveSettings, "RefractiveIndex"); + + Element dimensions = getFirstNode(image, "Dimensions"); + + Element tNode = getFirstNode(dimensions, "T"); + if (tNode != null) { + Element positions = getFirstNode(tNode, "Positions"); + if (positions != null) { + Element interval = getFirstNode(positions, "Interval"); + if (interval != null) { + Element incrementNode = getFirstNode(interval, "Increment"); + if (incrementNode != null) { + String increment = incrementNode.getTextContent(); + timeIncrement = new Time(DataTools.parseDouble(increment), UNITS.SECOND); + } } } } - } - Element sNode = getFirstNode(dimensions, "S"); - if (sNode != null) { - NodeList scenes = sNode.getElementsByTagName("Scene"); - int nextPosition = 0; - for (int i=0; i(); + + Element sNode = getFirstNode(dimensions, "S"); + if (sNode != null) { + NodeList scenes = sNode.getElementsByTagName("Scene"); + //int nextPosition = 0; + for (int i=0; i 0 && nextPosition < positionsX.length) { + + if (positions.getLength() == 0) {// && (mosaics <= 1 || (prestitched != null && prestitched))) { + positions = scene.getElementsByTagName("CenterPosition"); + //if (positions.getLength() > 0 && nextPosition < positionsX.length) { Element position = (Element) positions.item(0); String[] pos = position.getTextContent().split(","); - positionsX[nextPosition] = new Length(DataTools.parseDouble(pos[0]), UNITS.MICROMETER); - positionsY[nextPosition] = new Length(DataTools.parseDouble(pos[1]), UNITS.MICROMETER); + XYZLength loc = new XYZLength(); + + loc.pX = new Length(DataTools.parseDouble(pos[0]), UNITS.MICROMETER); + loc.pY = new Length(DataTools.parseDouble(pos[1]), UNITS.MICROMETER); + + currentSceneProps.pos.add(loc); + //} + //nextPosition++; } - nextPosition++; + } } - } - NodeList channelNodes = getGrandchildren(dimensions, "Channel"); - if (channelNodes == null) { - channelNodes = image.getElementsByTagName("Channel"); - } + NodeList channelNodes = getGrandchildren(dimensions, "Channel"); + if (channelNodes == null) { + channelNodes = image.getElementsByTagName("Channel"); + } - if (channelNodes != null) { - for (int i=0; i uniqueDetectors = new HashSet(); - for (int i=0; i uniqueDetectors = new HashSet<>(); + for (int i=0; i= gains.size()) { - store.setDetectorGain(DataTools.parseDouble(gain), 0, detectorIndex); + String detectorID = detector.getAttribute("Id"); + if (detectorID.indexOf(' ') != -1) { + detectorID = detectorID.replaceAll("\\s",""); } - else { - store.setDetectorGain( - DataTools.parseDouble(gains.get(detectorIndex)), 0, - detectorIndex); + if (!detectorID.startsWith("Detector:")) { + detectorID = "Detector:" + detectorID; } - } + if (uniqueDetectors.contains(detectorID)) { + continue; + } + uniqueDetectors.add(detectorID); + int detectorIndex = uniqueDetectors.size() - 1; - String offset = getFirstNodeValue(detector, "Offset"); - if (offset != null && !offset.equals("")) { - store.setDetectorOffset(new Double(offset), 0, detectorIndex); - } + store.setDetectorID(detectorID, 0, detectorIndex); + store.setDetectorManufacturer(manufacturer, 0, detectorIndex); + store.setDetectorModel(model, 0, detectorIndex); + store.setDetectorSerialNumber(serialNumber, 0, detectorIndex); + store.setDetectorLotNumber(lotNumber, 0, detectorIndex); - zoom = getFirstNodeValue(detector, "Zoom"); - if (zoom != null && !zoom.isEmpty()) { - if (zoom != null && !zoom.equals("")) { - if (zoom.indexOf(',') != -1) { - zoom = zoom.substring(0, zoom.indexOf(',')); + + gain = getFirstNodeValue(detector, "Gain"); + if (gain != null && !gain.isEmpty()) { + if (detectorIndex == 0 || detectorIndex >= gains.size()) { + store.setDetectorGain(DataTools.parseDouble(gain), 0, detectorIndex); + } + else { + store.setDetectorGain( + DataTools.parseDouble(gains.get(detectorIndex)), 0, + detectorIndex); } - store.setDetectorZoom(DataTools.parseDouble(zoom), 0, detectorIndex); } - } - String ampGain = getFirstNodeValue(detector, "AmplificationGain"); - if (ampGain != null && !ampGain.equals("")) { - store.setDetectorAmplificationGain(new Double(ampGain), 0, detectorIndex); - } + String offset = getFirstNodeValue(detector, "Offset"); + if (offset != null && !offset.equals("")) { + store.setDetectorOffset(Double.parseDouble(offset), 0, detectorIndex); + } + + zoom = getFirstNodeValue(detector, "Zoom"); + if (zoom != null && !zoom.isEmpty()) { + if (!zoom.equals("")) { + if (zoom.indexOf(',') != -1) { + zoom = zoom.substring(0, zoom.indexOf(',')); + } + store.setDetectorZoom(DataTools.parseDouble(zoom), 0, detectorIndex); + } + } + + String ampGain = getFirstNodeValue(detector, "AmplificationGain"); + if (ampGain != null && !ampGain.equals("")) { + store.setDetectorAmplificationGain(Double.parseDouble(ampGain), 0, detectorIndex); + } - String detectorType = getFirstNodeValue(detector, "Type"); - if (detectorType != null && !detectorType.equals("")) { - store.setDetectorType(MetadataTools.getDetectorType(detectorType), 0, detectorIndex); + String detectorType = getFirstNodeValue(detector, "Type"); + if (detectorType != null && !detectorType.equals("")) { + store.setDetectorType(MetadataTools.getDetectorType(detectorType), 0, detectorIndex); + } } } - } - NodeList objectives = getGrandchildren(instrument, "Objective"); - parseObjectives(objectives); + NodeList objectives = getGrandchildren(instrument, "Objective"); + parseObjectives(objectives); - NodeList filterSets = getGrandchildren(instrument, "FilterSet"); - if (filterSets != null) { - for (int i=0; i 0) { - store.setFilterSetDichroicRef(dichroicRef, 0, i); - } + if (dichroicRef != null && dichroicRef.length() > 0) { + store.setFilterSetDichroicRef(dichroicRef, 0, i); + } - if (excitations != null) { - for (int ex=0; ex 0) { + if (ref.length() > 0) { store.setFilterSetExcitationFilterRef(ref, 0, i, ex); } } - } - if (emissions != null) { - for (int em=0; em 0) { + if (ref.length() > 0) { store.setFilterSetEmissionFilterRef(ref, 0, i, em); } } } } - } - NodeList filters = getGrandchildren(instrument, "Filter"); - if (filters != null) { - for (int i=0; i 0) { + return nodes.item(0).getTextContent(); } + return null; + } - imageName = getFirstNodeValue(document, "Name"); + private static NodeList getGrandchildren(Element root, String name) { + return getGrandchildren(root, name + "s", name); + } + + private static NodeList getGrandchildren(Element root, String child, String name) { + if (root == null) { + return null; + } + NodeList children = root.getElementsByTagName(child); + if (children.getLength() > 0) { + Element childNode = (Element) children.item(0); + return childNode.getElementsByTagName(name); + } + return null; } - } - private void translateScaling(Element root) { - NodeList scalings = root.getElementsByTagName("Scaling"); - if (scalings == null || scalings.getLength() == 0) { - return; + private Element getFirstNode(Element root, String name) { + if (root == null) { + return null; + } + NodeList list = root.getElementsByTagName(name); + if (list.getLength() == 0) { + return null; + } + return (Element) list.item(0); } - Element scaling = (Element) scalings.item(0); - NodeList distances = getGrandchildren(scaling, "Items", "Distance"); + private void setAdditionalImageMetadata() throws FormatException { + + for (int iSeries=0; iSeries= 0) { continue; } - Double value = new Double(originalValue) * 1000000; - if (value > 0) { - PositiveFloat size = new PositiveFloat(value); - if (id.equals("X")) { - for (int series=0; series 0) { + reader.store.setImageDescription(description, iSeries); + } + + if (airPressure != null) { + reader.store.setImagingEnvironmentAirPressure( + new Pressure(Double.parseDouble(airPressure), UNITS.MILLIBAR), iSeries); + } + + if (co2Percent != null) { + reader.store.setImagingEnvironmentCO2Percent( + PercentFraction.valueOf(co2Percent), iSeries); + } + if (humidity != null) { + reader.store.setImagingEnvironmentHumidity( + PercentFraction.valueOf(humidity), iSeries); + } + if (temperature != null) { + reader.store.setImagingEnvironmentTemperature(new Temperature( + Double.valueOf(temperature), UNITS.CELSIUS), iSeries); + } + + if (objectiveSettingsID != null) { + reader.store.setObjectiveSettingsID(objectiveSettingsID, iSeries); + if (correctionCollar != null) { + reader.store.setObjectiveSettingsCorrectionCollar( + Double.parseDouble(correctionCollar), iSeries); + } + if (medium != null) { + reader.store.setObjectiveSettingsMedium(MetadataTools.getMedium(medium), iSeries); + } + if (refractiveIndex != null) { + reader.store.setObjectiveSettingsRefractiveIndex( + Double.parseDouble(refractiveIndex), iSeries); + } + } + } + } + + private void setImageNames() { + // Forces recomputing + reader.series = -1; + int nSeries = reader.getSeriesCount(); + String name = new Location(reader.getCurrentFile()).getName(); + if (imageName != null && imageName.trim().length() > 0) { + name = imageName; + } + + int indexLength = String.valueOf(nSeries).length(); + + for (int iSeries=0; iSeries>> + mapCoreCZTToBlocks, + DocumentBuilder parser) throws IOException { + CZTKey czt = new CZTKey(c,z,t); + List blocks = mapCoreCZTToBlocks.get(coreIdx).get(czt); + if ((blocks==null) || (blocks.size()==0)) return null; + LibCZI.SubBlockSegment block = LibCZI.getBlock(reader.getStream(blocks.get(0).filePart), blocks.get(0).filePosition); + return LibCZI.readSubBlockMeta(reader.getStream(blocks.get(0).filePart), block, parser); + } + + private class Corner { + Length x, y, z; + + Corner fromSubBlockMeta(Collection blocks, int iCoreIndex, Unit unitLength, DocumentBuilder parser) { + + for (MinDimEntry iBlock : blocks) { + try { + LibCZI.SubBlockSegment block = LibCZI.getBlock(reader.getStream(iBlock.filePart), iBlock.filePosition); + LibCZI.SubBlockMeta sbm = LibCZI.readSubBlockMeta(reader.getStream(iBlock.filePart), block, parser); + + if (sbm.stageX != null) { + // if true, the unit is micrometer + unitLength = UNITS.MICROMETER; + if ((x == null) || (x.value(unitLength).doubleValue() > sbm.stageX.value(unitLength).doubleValue())) { + x = new Length(sbm.stageX.value(unitLength).doubleValue(), unitLength); + } + } + + if (sbm.stageY != null) { + // if true, the unit is micrometer + unitLength = UNITS.MICROMETER; + if ((y == null) || (y.value(unitLength).doubleValue() > sbm.stageY.value(unitLength).doubleValue())) { + y = new Length(sbm.stageY.value(unitLength).doubleValue(), unitLength); + } + } + if ((z == null) || (z.value(unitLength).doubleValue() > sbm.stageZ.value(unitLength).doubleValue())) { + z = sbm.stageZ; } + } catch (Exception e) { + e.printStackTrace(); + return this; } - else if (id.equals("Z")) { - zStep = FormatTools.createLength(size, UNITS.MICROMETER); - for (int series=0; series blocks, int iCoreIndex, Unit unitLength) { + for (MinDimEntry iBlock : blocks) { + Length posX = new Length((double) iBlock.dimensionStartX/(double) reader.coreIndexToDownscaleFactor.get(iCoreIndex) + *coreToPixSizeX.get(iCoreIndex).value(unitLength).doubleValue(), unitLength); + if ((x == null)||(x.value().doubleValue()>posX.value(unitLength).doubleValue())) { + x = posX; + } + Length posY = new Length((double) iBlock.dimensionStartY/(double) reader.coreIndexToDownscaleFactor.get(iCoreIndex) + *coreToPixSizeY.get(iCoreIndex).value(unitLength).doubleValue(), unitLength); + if ((y == null)||(y.value().doubleValue()>posY.value(unitLength).doubleValue())) { + y = posY; + } + if (coreToPixSizeZ.size()!=0) { + if (coreToPixSizeZ.get(iCoreIndex).unit().equals(unitLength)) { // Sometimes, x and y are in um while z is undef + Length posZ = new Length((double) iBlock.dimensionStartZ + * coreToPixSizeZ.get(iCoreIndex).value(unitLength).doubleValue(), unitLength); + if ((z == null) || (z.value().doubleValue() > posZ.value(unitLength).doubleValue())) { + z = posZ; + } } } } - else { - LOGGER.debug( - "Expected positive value for PhysicalSize; got {}", value); + return this; + } + + } + + private void setSpaceAndTimeInformation( // of series and of planes + List< // CoreIndex + HashMap>> + mapCoreCZTToBlocks, + DocumentBuilder parser, CZISegments cziSegments) throws IOException { + + // Time : timestamp storing when each series was acquired + coreIndexTimeStamp = new Timestamp[reader.core.size()]; + + // Flags that prevents the reading of all metadata subblocks - there are way too many in LLS acquisition and + // this slows down drastically the initialisation time + boolean interpolatePlanePositionOverZ = isLatticeLightSheet(); + + // SPACE : according to czi specs, all subblocks are located within a common 2D virtual plane + // However, there's no easy way to know the physical coordinates of the origin of this virtual plane. + // See thread https://forum.image.sc/t/czi-plane-position-what-to-pick/85484 + // This reading: + // - uses the subblock coordinates to position each sub-block relative ot one another + // - attempts to find the physical coordinate of the virtual plane by several ways: + // - Strategy 1: computing the physical coordinate from a single sub-block which contain the stage position in the subblock metadata + // - OR Strategy 2: getting the physical coordinate from the main document metadata (first xml segment) + // - this offset is stored in the two variables below. If both methods fails, the corner pos is NaN + // and is subsequently ignored + double cornerXAllScenesMicrons = Double.NaN; + double cornerYAllScenesMicrons = Double.NaN; + + // Attempt strategy 1: + // - Let's pick the first subblock of the first core index + int coreUsedForXYOffset = 0; + MinDimEntry firstSubBlock = mapCoreCZTToBlocks.get(coreUsedForXYOffset).get(new CZTKey(0, 0, 0)).get(0); + + LibCZI.SubBlockSegment block = LibCZI.getBlock(reader.getStream(firstSubBlock.filePart), firstSubBlock.filePosition); + LibCZI.SubBlockMeta sbm = LibCZI.readSubBlockMeta(reader.getStream(firstSubBlock.filePart), block, parser); + // Here we assume the first core is not a special weird image like label or overview + // - Is there any valid metadata tags for the stage position ? + if (((sbm.stageX!=null)&&(sbm.stageX.unit().equals(UNITS.MICROMETER)))&& + ((sbm.stageY!=null)&&(sbm.stageY.unit().equals(UNITS.MICROMETER)))&& + (coreToPixSizeX.get(coreUsedForXYOffset).unit().equals(UNITS.MICROMETER))&& + (coreToPixSizeY.get(coreUsedForXYOffset).unit().equals(UNITS.MICROMETER))) { + // Yes! Let's apply Strategy 1 + // What is the block coordinates in pixel? + int coordX = firstSubBlock.dimensionStartX; + int coordY = firstSubBlock.dimensionStartY; + // Let's compute this block coordinates in micrometer + double pixSizeXMicrometer = coreToPixSizeX.get(coreUsedForXYOffset).value(UNITS.MICROMETER).doubleValue(); + double pixSizeYMicrometer = coreToPixSizeY.get(coreUsedForXYOffset).value(UNITS.MICROMETER).doubleValue(); + double coordXBlockMicrometer = coordX*pixSizeXMicrometer; + double coordYBlockMicrometer = coordY*pixSizeYMicrometer; + // Ok, now we have the correspondence between block dimension entry coordinates in micrometer: + // (coordXMiddleBlockMicrometer, coordYMiddleBlockMicrometer) + // and the subblock coordinates from the stage metadata: + double metaStageCoordX = sbm.stageX.value(UNITS.MICROMETER).doubleValue(); + double metaStageCoordY = sbm.stageY.value(UNITS.MICROMETER).doubleValue(); + // We can thus deduce the physical coordinates at the origin of the 2D virtual plane + cornerXAllScenesMicrons = metaStageCoordX-coordXBlockMicrometer; + cornerYAllScenesMicrons = metaStageCoordY-coordYBlockMicrometer; + LOGGER.debug("Global image origin X acquired from sub-block meta (micrometer) {}", cornerXAllScenesMicrons); + LOGGER.debug("Global image origin Y acquired from sub-block meta (micrometer) {}", cornerYAllScenesMicrons); + } else { + // Let's attempt Strategy 2 + if ((allPositionsInformation.scenes.size()>0)&&(!coreToPixSizeX.get(0).unit().equals(UNITS.REFERENCEFRAME))) { + // Get the coordinate of the most top left corner of all scenes + Length minX = null, minY = null; + for (SceneProperties scene:allPositionsInformation.scenes) { + Length scenePosX = scene.getMinPosXInMicrons(); + Length scenePosY = scene.getMinPosYInMicrons(); + if ((scenePosX!=null)&&(scenePosX.unit().equals(UNITS.MICROMETER))) { + if ((minX==null)||(minX.value(UNITS.MICROMETER).doubleValue()>scenePosX.value(UNITS.MICROMETER).doubleValue())) { + minX = scenePosX; + } + } + if ((scenePosY!=null)&&(scenePosY.unit().equals(UNITS.MICROMETER))) { + if ((minY==null)||(minY.value(UNITS.MICROMETER).doubleValue()>scenePosY.value(UNITS.MICROMETER).doubleValue())) { + minY = scenePosY; + } + } + } + if ((minX!=null)&&(minY!=null)) { + // Applies strategy 2 + cornerXAllScenesMicrons = minX.value(UNITS.MICROMETER).doubleValue(); + cornerYAllScenesMicrons = minY.value(UNITS.MICROMETER).doubleValue(); + LOGGER.debug("Global image origin X acquired from global meta (micrometer) {}", cornerXAllScenesMicrons); + LOGGER.debug("Global image origin Y acquired from global meta (micrometer) {}", cornerYAllScenesMicrons); + } } } - } - } - private void translateDisplaySettings(Element root) throws FormatException { - NodeList displaySettings = root.getElementsByTagName("DisplaySetting"); - if (displaySettings == null || displaySettings.getLength() == 0) { - return; - } + Map timeStampsResolutionLevel0 = new HashMap<>(); + Length planePosXResolutionLevel0 = null; + Length planePosYResolutionLevel0 = null; + Length planePosZResolutionLevel0 = null; - for (int display=0; display= 0) { + continue; + } + + // Set properly series and corresponding core index + // (the setCoreIndex methods behaves weirdly in the super class, that's why it is set as shown below) + reader.coreIndex = iCoreIndex; + reader.series = reader.coreIndexToSeries.get(reader.coreIndex); + + // A flag that's useful to know if the current core corresponds to a lower resolution level + // useful because the metadata for lower resolution levels will be copied from the highest one + boolean resolutionLevel0 = reader.coreIndexToDownscaleFactor.get(reader.coreIndex)==1; + + // This unit will be either ReferenceFrame for some czi images (for instance artificial dataset + // not generated from a microscope) or, micrometers in the other cases + Unit unitLength = coreToPixSizeX.get(iCoreIndex).unit(); + + CoreSignature signature = reader.coreIndexToSignature.get(iCoreIndex); + + // Set stage name + String sceneName; + Length scenePosZ = null; + if (signature.getDimensions().containsKey("S")) { + int sceneIndex = signature.getDimensions().get("S"); + if (allPositionsInformation.scenes.size()>sceneIndex) { + sceneName = allPositionsInformation.scenes.get(sceneIndex).name; + if ((sceneName == null)||(sceneName.trim().equals(""))) sceneName = "Scene position #"+sceneIndex; + scenePosZ = allPositionsInformation.scenes.get(sceneIndex).getMinPosZInMicrons(); + } else { + sceneName = "Scene position #"+sceneIndex; + } + } else { + // default name + sceneName = "Scene position #"+0; + if (allPositionsInformation.scenes.size()==1) { + scenePosZ = allPositionsInformation.scenes.get(0).getMinPosZInMicrons(); + } + } + reader.store.setStageLabelName(sceneName, reader.series); + + // TIME: Let's set the acquisition date of this series : it's the same + // for all series, but there will be an offset on the first plane + + Timestamp seriesT0 = null; + if (acquiredDate!=null) { + seriesT0 = new Timestamp(acquiredDate); + reader.store.setImageAcquisitionDate(seriesT0, reader.series); + } + + int nChannels = reader.getSizeC(); + List blocks; + + Length planePosX, planePosY, planePosZ = null; // plane position of the current coreindex - do not vary over z and t, but that could happen + + blocks = mapCoreCZTToBlocks.get(iCoreIndex).get(new CZTKey(0,0,0)); + + if (!resolutionLevel0) { + // Use the same position as the higher resolution level + // Keeping the last highest resolutio works because bio-formats forces the resolution + // level series to be sorted according to the core series index: + // res 0 series i / res 1 series i / res 2 series i / res 0 series i+1 / res 1 series i+1 etc. + planePosX = planePosXResolutionLevel0; + planePosY = planePosYResolutionLevel0; + planePosZ = planePosZResolutionLevel0; + } else { + // The most complicated logic + // ------ Here's the available data: + + // XYZ - Plane position according to the block position (should not be null - compulsory in czi file) + Corner blocksCorner = new Corner().fromBlocks(blocks, iCoreIndex, unitLength); + // XYZ - Plane position according to the subblock xml metadata (could be null) + Corner subBlockMetaCorner = new Corner().fromSubBlockMeta(blocks, iCoreIndex, unitLength, parser); + // XY - (offsetXInMicrons, offsetYInMicrons) : the top left corner, according to the xml czi header (could be NaN) + // Z - scenePosZ : the z location of the current scene, according to the xml czi header (could be Null) + /*System.out.println("- Id = "+reader.currentId); + System.out.println("- CoreIndex = "+reader.coreIndex+" Res Lev 0 = "+resolutionLevel0); + System.out.println("Min XYZ location according to block start position:"); + System.out.println("\tX="+blocksCorner.x); + System.out.println("\tY="+blocksCorner.y); + System.out.println("\tZ="+blocksCorner.z); + System.out.println("Min XYZ location according to subblock metadata position:"); + System.out.println("\tX="+subBlockMetaCorner.x); + System.out.println("\tY="+subBlockMetaCorner.y); + System.out.println("\tZ="+subBlockMetaCorner.z); + System.out.println("XY Global offset according to main xml document:"); + System.out.println("\tX="+offsetXInMicrons); + System.out.println("\tY="+offsetYInMicrons); + System.out.println("Z position of current scene according to main xml document:"); + System.out.println("\tZ="+scenePosZ);*/ + + // XY : keep blocks only + planePosX = blocksCorner.x; + planePosY = blocksCorner.y; + + if (planePosX.unit().equals(UNITS.MICROMETER) && (planePosY.unit().equals(UNITS.MICROMETER))) { + // Let's add a global stage offset, if possible + if ((!Double.isNaN(cornerXAllScenesMicrons))&&(!Double.isNaN(cornerXAllScenesMicrons))) { + planePosX = new Length(planePosX.value(UNITS.MICROMETER).doubleValue()+cornerXAllScenesMicrons, UNITS.MICROMETER); + planePosY = new Length(planePosY.value(UNITS.MICROMETER).doubleValue()+cornerYAllScenesMicrons, UNITS.MICROMETER); + } } - while (channels.size() <= i) { - channels.add(new Channel()); + if ((blocksCorner.z!=null)&&(subBlockMetaCorner.z!=null)&&(blocksCorner.z.unit().equals(subBlockMetaCorner.z.unit()))) { + Unit u = blocksCorner.z.unit(); + planePosZ = new Length(blocksCorner.z.value(u).doubleValue()+subBlockMetaCorner.z.value(u).doubleValue(), u); + } else if ((blocksCorner.z!=null)&&(scenePosZ!=null)&&(scenePosZ.unit().equals(blocksCorner.z.unit()))) { + Unit u = blocksCorner.z.unit(); + planePosZ = new Length(blocksCorner.z.value(u).doubleValue()+scenePosZ.value(u).doubleValue(), u); + } else if (blocksCorner.z!=null) { + planePosZ = blocksCorner.z; + } else if (subBlockMetaCorner.z!=null) { + planePosZ = subBlockMetaCorner.z; + } else if (scenePosZ!=null) { + planePosZ = scenePosZ; + } + + } + + if (resolutionLevel0) { + planePosXResolutionLevel0 = planePosX; + planePosYResolutionLevel0 = planePosY; + planePosZResolutionLevel0 = planePosZ; + } + + loopChannel: + for (int iChannel = 0; iChannel1) { + incrementTimeOverZ = (sbmzfti.timestamp - sbmziti.timestamp) / (double) ((reader.getSizeZ()-1) / reader.nRotations); } - String name = channel.getAttribute("Name"); - if (name != null) { - channels.get(i).name = name; + double incrementTimeOverT = 0; + if (reader.getSizeT()>1) { + incrementTimeOverT = (sbmzitf.timestamp - sbmziti.timestamp) / (double) ((reader.getSizeT()-1) / reader.nPhases); + } + Time exposure = null; + if (!Double.isNaN(sbmziti.exposureTime)) { + exposure = new Time(sbmziti.exposureTime*1000, UNITS.SECOND); + } else { + if (channels.size()>iChannel) { // Could be an issue with illumination TODO check + Double exposureFromChannel = channels.get(iChannel).exposure; + if (exposureFromChannel != null) { + exposure = new Time(exposureFromChannel, UNITS.SECOND); + } + } } - String emission = getFirstNodeValue(channel, "DyeMaxEmission"); - if (emission != null) { - channels.get(i).emission = emission; + double offsetT0; + double realT0 = 0; + if (!Double.isNaN(sbmziti.timestamp)) { + if (seriesT0 == null) { + offsetT0 = sbmziti.timestamp; + } else { + realT0 = seriesT0.asInstant().getMillis() / 1000.0; + offsetT0 = sbmziti.timestamp - realT0; + } + } else { + if (cziSegments.timeStamps.length>0) { + offsetT0 = cziSegments.timeStamps[0]; // In seconds + realT0 = offsetT0; + } else { + offsetT0 = 0; + } } - String excitation = getFirstNodeValue(channel, "DyeMaxExcitation"); - if (excitation != null) { - channels.get(i).excitation = excitation; + double offsetZ0; + if (planePosZ==null) { + offsetZ0 = Double.NaN; + } else { + if (planePosZ.value(unitLength) == null) { + offsetZ0 = Double.NaN; + } else { + offsetZ0 = planePosZ.value(unitLength).doubleValue(); + } } + double stepZ = (zStep==null)?0:zStep.value(unitLength).doubleValue(); + + if (resolutionLevel0) timeStampsResolutionLevel0.put(iChannel, new double[reader.getSizeT()*reader.getSizeZ()]); // Store the timestamps to be copied over resolution levels + double[] currentTimeStamps = timeStampsResolutionLevel0.get(iChannel); + int planeCounter = 0; + if ((reader.flattenedResolutions)||(resolutionLevel0)) { // Avoid setting the metadata not for lower resolution levels, because this override proper metadata if flattenresolution = false + for (int iTori = 0; iTori < reader.getSizeT(); iTori++) { + int iT = iTori % (reader.getSizeT() / reader.nPhases); + double timeStartCurrentFrame = 0; + if (interpolatePlanePositionOverZ) { + // We can afford to read the first block of the first timepoint, even for lattice + LibCZI.SubBlockMeta sbmzt = getSubBlockMeta( + iChannel, + 0, + iT, + iCoreIndex, mapCoreCZTToBlocks, parser); + if (Double.isNaN(sbmzt.timestamp)) { + if (cziSegments.timeStamps.length>iT) { + timeStartCurrentFrame = cziSegments.timeStamps[iT]; + } + } else { + timeStartCurrentFrame = sbmzt.timestamp - realT0; + } + } - String illumination = getFirstNodeValue(channel, "IlluminationType"); + for (int iZori = 0; iZori < reader.getSizeZ(); iZori++) { - if (illumination != null && (channels.get(i).illumination == null || channels.get(i).illumination == IlluminationType.OTHER)) { - channels.get(i).illumination = MetadataTools.getIlluminationType(illumination); + + int planeIndex = reader.getIndex(iZori, iChannel, iTori); + // rotations -> modulo Z + // illuminations -> modulo C + // phases -> modulo T + + reader.store.setPlanePositionX(planePosX, reader.series, planeIndex); + reader.store.setPlanePositionY(planePosY, reader.series, planeIndex); + + if ((exposure != null)&&(!Double.isNaN(exposure.value().doubleValue()))) { + reader.store.setPlaneExposureTime(exposure, reader.series, planeIndex); // 0 exposure do not make sense + } + int iZ = iZori % (reader.getSizeZ() / reader.nRotations); + + Time dT = null; + Length pZ = null; + + if (interpolatePlanePositionOverZ) { + if ((incrementTimeOverZ >= 0) || (incrementTimeOverT >= 0)) { + dT = new Time(timeStartCurrentFrame + incrementTimeOverZ * iZ, + UNITS.SECOND); + } + pZ = new Length(offsetZ0 + iZ * stepZ, unitLength); + } else { + if (resolutionLevel0) { + LibCZI.SubBlockMeta sbmzt = getSubBlockMeta( + iChannel, + iZ, + iT, + iCoreIndex, mapCoreCZTToBlocks, parser); + if (Double.isNaN(sbmzt.timestamp)) { + if (cziSegments.timeStamps.length>iT) { + dT = new Time(cziSegments.timeStamps[iT], + UNITS.SECOND); + } + } else { + dT = new Time(sbmzt.timestamp - realT0, + UNITS.SECOND); + } + } + + pZ = new Length(offsetZ0 + iZ * stepZ, unitLength); + } + if (resolutionLevel0) { + if ((dT != null) && (!Double.isNaN(dT.value().doubleValue()))) { // To fit the original reader. NaN -> Null + reader.store.setPlaneDeltaT(dT, reader.series, planeIndex); + currentTimeStamps[planeCounter] = dT.value().doubleValue(); + } + } else { + if ((!Double.isNaN(currentTimeStamps[planeCounter]))) { // To fit the original reader. NaN -> Null + reader.store.setPlaneDeltaT(new Time(currentTimeStamps[planeCounter], UNITS.SECOND), reader.series, planeIndex); + } + } + + if ((!Double.isNaN(offsetZ0))&&(pZ!=null)&&(!pZ.unit().equals(UNITS.REFERENCEFRAME))) { + reader.store.setPlanePositionZ(pZ, reader.series, planeIndex); + } + + if (planeIndex==0) { + reader.store.setStageLabelX(planePosX, reader.series); + reader.store.setStageLabelY(planePosY, reader.series); + if ((pZ!=null)&&(!Double.isNaN(pZ.value().doubleValue()))&&(!pZ.unit().equals(UNITS.REFERENCEFRAME))) { + reader.store.setStageLabelZ(pZ, reader.series); + } + } + planeCounter++; + } + } } } } + + for (int iCoreIndex = 0; iCoreIndex 0) { + for (int iCoreIndex=0; iCoreIndex= 0) continue; + + // setCoreIndex(iCoreIndex); -> THIS JUST DOES NOT SET THE RIGHT RESOLUTION!! TODO: Post an issue + // Hence this hack: + int downscale = reader.coreIndexToDownscaleFactor.get(iCoreIndex); + boolean resolutionZero = (downscale==1)||(reader.flattenedResolutions);// The OR test is there because you need to fill all series with a pixel size sequentially. + + PositiveFloat size = new PositiveFloat(value); + + switch (id) { + case "X": + coreToPixSizeX.put(iCoreIndex, FormatTools.createLength(size.getValue() * downscale, UNITS.MICROMETER)); + if (resolutionZero) { + reader.store.setPixelsPhysicalSizeX(coreToPixSizeX.get(iCoreIndex), reader.series); + } break; + case "Y": + coreToPixSizeY.put(iCoreIndex, FormatTools.createLength(size.getValue() * downscale, UNITS.MICROMETER)); + if (resolutionZero) { + reader.store.setPixelsPhysicalSizeY(coreToPixSizeY.get(iCoreIndex), reader.series); + } break; + case "Z": + zStep = FormatTools.createLength(size, UNITS.MICROMETER); + coreToPixSizeZ.put(iCoreIndex, zStep); + if (resolutionZero) { + reader.store.setPixelsPhysicalSizeZ(zStep, reader.series); + } break; + } + } + } + else { + LOGGER.debug( + "Expected positive value for PhysicalSize; got {}", value); + for (int iCoreIndex=0; iCoreIndex 0) { String roiID = MetadataTools.createLSID("ROI", roiCount); - store.setROIID(roiID, roiCount); - store.setROIName(layer.getAttribute("Name"), roiCount); - store.setROIDescription(getFirstNodeValue(layer, "Usage"), roiCount); + reader.store.setROIID(roiID, roiCount); + reader.store.setROIName(layer.getAttribute("Name"), roiCount); + reader.store.setROIDescription(getFirstNodeValue(layer, "Usage"), roiCount); - for (int series=0; series(); - if (groups != null) { - for (int i=0; i(); + for (int r=0; r 0) { - for (int i=0; i 0) { - for (int i=0; i 0) { - Element childNode = (Element) children.item(0); - return childNode.getElementsByTagName(name); - } - return null; - } - - private String getFirstNodeValue(Element root, String name) { - if (root == null) { - return null; - } - NodeList nodes = root.getElementsByTagName(name); - if (nodes != null && nodes.getLength() > 0) { - return nodes.item(0).getTextContent(); - } - return null; - } - - private void populateOriginalMetadata(Element root, Deque nameStack) { - String name = root.getNodeName(); - nameStack.push(name); - - final StringBuilder key = new StringBuilder(); - String k = null; - Iterator keys = nameStack.descendingIterator(); - while (keys.hasNext()) { - k = keys.next(); - if (!k.equals("Metadata") && (!k.endsWith("s") || k.equals(name))) { - key.append(k); - key.append("|"); - } - } - - if (root.getChildNodes().getLength() == 1) { - String value = root.getTextContent(); - if (value != null && key.length() > 0) { - String s = key.toString(); - if (s.endsWith("|")){ - s = s.substring(0, s.length() - 1); - } - if (s.startsWith("DisplaySetting")) { - addGlobalMeta(s, value); - } - else { - addGlobalMetaList(s, value); - } - - if (key.toString().endsWith("|Rotations|")) { - rotationLabels = value.split(" "); - } - else if (key.toString().endsWith("|Phases|")) { - phaseLabels = value.split(" "); - } - else if (key.toString().endsWith("|Illuminations|")) { - illuminationLabels = value.split(" "); - } - } - } - NamedNodeMap attributes = root.getAttributes(); - for (int i=0; i 6) { + color = color.substring(2); + } + try { + // shift by 8 to allow alpha in the final byte + reader.store.setChannelColor( + new Color((Integer.parseInt(color, 16) << 8) | 0xff), iSeries, c); + } + catch (NumberFormatException e) { + LOGGER.warn("", e); + } + } - if (skipData && !(segment instanceof Attachment)) { - segment.close(); - return null; - } - return segment; - } + String emWave = channels.get(c).emission; + if (emWave != null) { + Double wave = Double.parseDouble(emWave); + Length em = FormatTools.getEmissionWavelength(wave); + if (em != null) { + reader.store.setChannelEmissionWavelength(em, iSeries, c); + } + } + String exWave = channels.get(c).excitation; + if (exWave != null) { + Double wave = Double.valueOf(exWave); + Length ex = FormatTools.getExcitationWavelength(wave); + if (ex != null) { + reader.store.setChannelExcitationWavelength(ex, iSeries, c); + } + } - private void convertPixelType(int pixelType) throws FormatException { - CoreMetadata ms0 = core.get(0); - convertPixelType(ms0, pixelType); - } + if (channels.get(c).illumination != null) { + reader.store.setChannelIlluminationType( + channels.get(c).illumination, iSeries, c); + } - private void convertPixelType(CoreMetadata ms0, int pixelType) throws FormatException { - switch (pixelType) { - case GRAY8: - ms0.pixelType = FormatTools.UINT8; - break; - case GRAY16: - ms0.pixelType = FormatTools.UINT16; - break; - case GRAY32: - ms0.pixelType = FormatTools.UINT32; - break; - case GRAY_FLOAT: - ms0.pixelType = FormatTools.FLOAT; - break; - case GRAY_DOUBLE: - ms0.pixelType = FormatTools.DOUBLE; - break; - case BGR_24: - ms0.pixelType = FormatTools.UINT8; - ms0.sizeC *= 3; - ms0.rgb = true; - ms0.interleaved = true; - break; - case BGR_48: - ms0.pixelType = FormatTools.UINT16; - ms0.sizeC *= 3; - ms0.rgb = true; - ms0.interleaved = true; - break; - case BGRA_8: - ms0.pixelType = FormatTools.UINT8; - ms0.sizeC *= 4; - ms0.rgb = true; - ms0.interleaved = true; - break; - case BGR_FLOAT: - ms0.pixelType = FormatTools.FLOAT; - ms0.sizeC *= 3; - ms0.rgb = true; - ms0.interleaved = true; - break; - case COMPLEX: - case COMPLEX_FLOAT: - throw new FormatException("Sorry, complex pixel data not supported."); - default: - throw new FormatException("Unknown pixel type: " + pixelType); - } - ms0.interleaved = ms0.rgb; - } + if (channels.get(c).pinhole != null) { + reader.store.setChannelPinholeSize( + new Length(Double.valueOf(channels.get(c).pinhole), UNITS.MICROMETER), iSeries, c); + } - private void parseObjectives(NodeList objectives) throws FormatException { - if (objectives != null) { - for (int i=0; i= 0) { + continue; } - } - } - - public void close() throws IOException { - // whatever created the Segment is responsible for closing the stream - // we just need to remove the reference - stream = null; - } - public RandomAccessInputStream getStream() throws IOException { - if (stream != null) { - return stream; + boolean isPALM = false; // TODO + addChannelMetadata(iSeries, isPALM); } - return new RandomAccessInputStream(filename, BUFFER_SIZE); } - } - - /** Segment with ID "ZISRAWFILE". */ - class FileHeader extends Segment { - public int majorVersion; - public int minorVersion; - public long primaryFileGUID; - public long fileGUID; - public int filePart; - public long directoryPosition; - public long metadataPosition; - public boolean updatePending; - public long attachmentDirectoryPosition; - - @Override - public void fillInData() throws IOException { - super.fillInData(); - RandomAccessInputStream s = getStream(); - try { - s.order(isLittleEndian()); - s.seek(startingPosition + HEADER_SIZE); - majorVersion = s.readInt(); - minorVersion = s.readInt(); - s.skipBytes(4); // reserved 1 - s.skipBytes(4); // reserved 2 - primaryFileGUID = s.readLong(); // 16 - fileGUID = s.readLong(); // 16 - filePart = s.readInt(); - - directoryPosition = s.readLong(); - metadataPosition = s.readLong(); - updatePending = s.readInt() != 0; - attachmentDirectoryPosition = s.readLong(); - } - finally { - if (stream == null) { - s.close(); - } - } - } - } + private void parseObjectives(NodeList objectives) throws FormatException { + if (objectives != null) { + for (int i=0; i= 0) { - public byte[] readPixelData(RandomAccessInputStream s, Region tile, byte[] buf) throws FormatException, IOException { - s.order(isLittleEndian()); - s.seek(dataOffset); + int iSeries = coreIndexToSeries(iCoreIndex); + if (extraIndex == 0) { + // Label Image + if (allPositionsInformation.labelPixelSize!=null) { + store.setPixelsPhysicalSizeX(allPositionsInformation.labelPixelSize.pX,iSeries); + store.setPixelsPhysicalSizeY(allPositionsInformation.labelPixelSize.pY,iSeries); + store.setPixelsPhysicalSizeZ(allPositionsInformation.labelPixelSize.pZ,iSeries); + } + if (allPositionsInformation.labelLocation!=null) { + store.setPlanePositionX(allPositionsInformation.labelLocation.pX,iSeries,0); + store.setPlanePositionY(allPositionsInformation.labelLocation.pY,iSeries,0); + store.setPlanePositionZ(allPositionsInformation.labelLocation.pZ,iSeries,0); + } + } else if (extraIndex == 1) { + // Macro Image + if (allPositionsInformation.slidePreviewPixelSize!=null) { + store.setPixelsPhysicalSizeX(allPositionsInformation.slidePreviewPixelSize.pX,iSeries); + store.setPixelsPhysicalSizeY(allPositionsInformation.slidePreviewPixelSize.pY,iSeries); + store.setPixelsPhysicalSizeZ(allPositionsInformation.slidePreviewPixelSize.pZ,iSeries); + } + if (allPositionsInformation.slidePreviewLocation!=null) { + store.setPlanePositionX(allPositionsInformation.slidePreviewLocation.pX,iSeries,0); + store.setPlanePositionY(allPositionsInformation.slidePreviewLocation.pY,iSeries,0); + store.setPlanePositionZ(allPositionsInformation.slidePreviewLocation.pZ,iSeries,0); + } + } + } + } + }*/ - if (directoryEntry.compression == UNCOMPRESSED) { - if (buf == null) { - buf = new byte[(int) dataSize]; - } - if (tile != null) { - readPlane(s, tile.x, tile.y, tile.width, tile.height, buf); - } - else { - s.readFully(buf); + private void populateOriginalMetadata(Element root, Deque nameStack) { + String name = root.getNodeName(); + nameStack.push(name); + + final StringBuilder key = new StringBuilder(); + String k; + Iterator keys = nameStack.descendingIterator(); + while (keys.hasNext()) { + k = keys.next(); + if (!k.equals("Metadata") && (!k.endsWith("s") || k.equals(name))) { + key.append(k); + key.append("|"); } - return buf; } - byte[] data = new byte[(int) dataSize]; - s.read(data); - - int bytesPerPixel = FormatTools.getBytesPerPixel(getPixelType()); - CodecOptions options = new CodecOptions(); - options.interleaved = isInterleaved(); - options.littleEndian = isLittleEndian(); - options.bitsPerSample = bytesPerPixel * 8; - options.maxBytes = - getSizeX() * getSizeY() * getRGBChannelCount() * bytesPerPixel; - - switch (directoryEntry.compression) { - case JPEG: - data = new JPEGCodec().decompress(data, options); - break; - case LZW: - data = new LZWCodec().decompress(data, options); - break; - case JPEGXR: - options.width = directoryEntry.dimensionEntries[0].storedSize; - options.height = directoryEntry.dimensionEntries[1].storedSize; - options.maxBytes = options.width * options.height * - getRGBChannelCount() * bytesPerPixel; - try { - data = new JPEGXRCodec().decompress(data, options); + if (root.getChildNodes().getLength() == 1) { + String value = root.getTextContent(); + if (value != null && key.length() > 0) { + String s = key.toString(); + if (s.endsWith("|")){ + s = s.substring(0, s.length() - 1); } - catch (FormatException e) { - if (data.length == options.maxBytes) { - LOGGER.debug("Invalid JPEG-XR compression flag"); - } - else { - LOGGER.warn("Could not decompress block; some pixels may be 0", e); - data = new byte[options.maxBytes]; - } + if (s.startsWith("DisplaySetting")) { + reader.addGlobalMeta(s, value); } - break; - case ZSTD_0: - data = new ZstdCodec().decompress(data); - break; - case ZSTD_1: - boolean highLowUnpacking = false; - int pointer = 0; - try (RandomAccessInputStream stream = new RandomAccessInputStream(data)) { - int sizeOfHeader = readVarint(stream); - while (stream.getFilePointer() < sizeOfHeader) { - int chunkID = readVarint(stream); - // only one chunk ID defined so far - switch (chunkID) { - case 1: - int payload = stream.read(); - highLowUnpacking = (payload & 1) == 1; - break; - default: - throw new FormatException("Invalid chunk ID: " + chunkID); - } - } - // safe cast because stream wraps a byte array - pointer = (int) stream.getFilePointer(); + else { + reader.addGlobalMetaList(s, value); } - byte[] decoded = new ZstdCodec().decompress(data, pointer, data.length - pointer); - // ZSTD_1 implies high/low byte unpacking, so it would be weird - // if this flag were unset - if (highLowUnpacking) { - data = new byte[decoded.length]; - int secondHalf = decoded.length / 2; - for (int i=0; i= data.length) { - System.arraycopy(data, 0, buf, 0, data.length); - return buf; + } } - return data; - } + NamedNodeMap attributes = root.getAttributes(); + for (int i=0; i pos = new ArrayList<>(); + String name; - @Override - public void fillInData() throws IOException { - super.fillInData(); - - RandomAccessInputStream s = getStream(); - try { - s.order(isLittleEndian()); - s.seek(startingPosition + HEADER_SIZE); - - int entryCount = s.readInt(); - s.skipBytes(124); - entries = new DirectoryEntry[entryCount]; - for (int i=0; i 1) { - dimensionEntries[i] = null; - if (i > 0) { - dimensionEntries[i - 1] = null; + for (XYZLength positionLocation : pos) { + if ((positionLocation.pZ==null)||(positionLocation.pZ.value()==null)) { + return null; } - } - } - } - - public DimensionEntry getDimensionEntry(String dimension) { - if (dimension != null) { - for (DimensionEntry entry : dimensionEntries) { - if (entry.dimension != null && entry.dimension.equals(dimension)) { - return entry; + if (positionLocation.pZ.value().doubleValue() tiles = new ArrayList<>(); - static class DimensionEntry { - public String dimension; - public int start; - public int size; - public float startCoordinate; - public int storedSize; - - public DimensionEntry(RandomAccessInputStream s) throws IOException { - dimension = s.readString(4).trim(); - start = s.readInt(); - size = s.readInt(); - startCoordinate = s.readFloat(); - storedSize = s.readInt(); + int nTilesX, nTilesY; } - @Override - public String toString() { - return "dimension=" + dimension + ", start=" + start + ", size=" + size + - ", startCoordinate=" + startCoordinate + ", storedSize=" + storedSize; + private static class TileProperties { + final XYZLength pos = new XYZLength(); + Integer iX; + Integer iY; } - } - static class AttachmentEntry { - public String schemaType; - public long filePosition; - public int filePart; - public String contentGUID; - public String contentFileType; - public String name; - - public AttachmentEntry(RandomAccessInputStream s) throws IOException { - schemaType = s.readString(2); - s.skipBytes(10); // reserved - filePosition = s.readLong(); - filePart = s.readInt(); - contentGUID = s.readString(16); - contentFileType = s.readString(8); - name = s.readString(80); + private static class XYZLength { + Length pX, pY, pZ; } - @Override - public String toString() { - return "schemaType = " + schemaType + ", filePosition = " + filePosition + - ", filePart = " + filePart + ", contentGUID = " + contentGUID + - ", contentFileType = " + contentFileType; - } - } + private static class AllPositionsInformation { + /** + * We may have, group, tiles, or scenes + */ + List scenes; + List groups; + //List tiles; - static class Channel { - public String name; - public String color; - public IlluminationType illumination; - public AcquisitionMode acquisitionMode; - public String emission; - public String excitation; - public String pinhole; - public Double exposure; - public Double gain; - public String fluor; - public String filterSetRef; - } + List regions; - static class Coordinate { - public int series; - public int plane; - private int imageCount; + XYZLength slidePreviewPixelSize, slidePreviewLocation, labelPixelSize, labelLocation; - public Coordinate(int series, int plane, int imageCount) { - this.series = series; - this.plane = plane; - this.imageCount = imageCount; } - @Override - public boolean equals(Object o) { - if (o == null || !(o instanceof Coordinate)) { - return false; - } - return ((Coordinate) o).series == this.series && - ((Coordinate) o).plane == this.plane; + private boolean isLatticeLightSheet = false; + private void setLatticeLightSheet(boolean flag) { + isLatticeLightSheet = flag; } - @Override - public int hashCode() { - return series * imageCount + plane; + private boolean isLatticeLightSheet() { + return isLatticeLightSheet; } - @Override - public String toString() { - return "[series = " + series + ", plane = " + plane + "]"; - } } } diff --git a/components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java b/components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java new file mode 100644 index 00000000000..3ba2d7e50fd --- /dev/null +++ b/components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java @@ -0,0 +1,1113 @@ +package loci.formats.in.libczi; + +import loci.common.Constants; +import loci.common.RandomAccessInputStream; +import loci.formats.in.ZeissCZIReader; +import ome.units.UNITS; +import ome.units.quantity.Length; +import ome.xml.model.primitives.Timestamp; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.w3c.dom.Element; +import org.w3c.dom.NodeList; +import org.xml.sax.SAXException; + +import javax.xml.parsers.DocumentBuilder; +import java.io.ByteArrayInputStream; +import java.io.IOException; + +/** + * CziStructs.h c++ structures translated to Java + * And helper static methods + *

+ * Translation: + * - using Java variable convention (lowercase) + * - std::int32_t is int + * - std::int64_t is long + * - GUID is 16 bytes long (unused) + * - char arrays are String + *

+ * Timestamps reading inspired by Jerome's macro + *

+ * See @see CZI reference documentation + *

+ * Used in {@link ZeissCZIReader} + * + * @author Nicolas Chiaruttini, EPFL, 2023 + */ +public class LibCZI { + + private static final Logger logger = LoggerFactory.getLogger(LibCZI.class); + + // --------------------------- PUBLIC HELPER METHODS + + /** + * @param id a czi file path + * @param BUFFER_SIZE the size of the caching buffer in bytes + * @param isLittleEndian endianness of the data + * @return the file header segment + * @throws IOException invalid file, invalid file location, file header not found + */ + public static FileHeaderSegment getFileHeaderSegment(String id, int BUFFER_SIZE, boolean isLittleEndian) throws IOException { + try (RandomAccessInputStream in = new RandomAccessInputStream(id, BUFFER_SIZE)) { + in.order(isLittleEndian); + int skip = + (ALIGNMENT - (int) (in.getFilePointer() % ALIGNMENT)) % ALIGNMENT; + in.skipBytes(skip); + long startingPosition = in.getFilePointer(); + String segmentID = in.readString(16).trim(); + if (segmentID.equals(ZISRAWFILE)) { + // That's correct, it's a CZI fileheader + FileHeaderSegment fileHeaderSegment = new FileHeaderSegment(); + // read the segment header + fileHeaderSegment.header.id = segmentID; + fileHeaderSegment.header.allocatedSize = in.readLong(); + fileHeaderSegment.header.usedSize = in.readLong(); + + fileHeaderSegment.data.major = in.readInt(); + fileHeaderSegment.data.minor = in.readInt(); + fileHeaderSegment.data._reserved1 = in.readInt(); + fileHeaderSegment.data._reserved2 = in.readInt(); + in.read(fileHeaderSegment.data.primaryFileGuid.bytes); + in.read(fileHeaderSegment.data.fileGuid.bytes); + fileHeaderSegment.data.filePart = in.readInt(); + fileHeaderSegment.data.subBlockDirectoryPosition = in.readLong(); + fileHeaderSegment.data.metadataPosition = in.readLong(); + fileHeaderSegment.data.updatePending = in.readInt(); + fileHeaderSegment.data.attachmentDirectoryPosition = in.readLong(); + return fileHeaderSegment; + } else { + throw new IOException(ZISRAWFILE+" segment expected, found "+segmentID+" instead."); + } + } + } + + /** + * @param fileHeader file header segment of the referenced czi file + * @param id a czi file path + * @param BUFFER_SIZE the size of the caching buffer in bytes + * @param isLittleEndian endianness of the data + * @return the subblock directory segment of the czi file + * @throws IOException invalid file, invalid file location, segment not found + */ + public static SubBlockDirectorySegment getSubBlockDirectorySegment(FileHeaderSegment fileHeader, String id, int BUFFER_SIZE, boolean isLittleEndian) throws IOException { + // TODO (maybe) : increase buffer size to limit the number of IO calls, especially when the file is mounted on a network drive + try (RandomAccessInputStream in = new RandomAccessInputStream(id, BUFFER_SIZE)) { + SubBlockDirectorySegment directorySegment = new SubBlockDirectorySegment(); + + in.order(isLittleEndian); + in.seek(fileHeader.data.subBlockDirectoryPosition); + + String segmentID = in.readString(16).trim(); + if (segmentID.equals(ZISRAWDIRECTORY)) { + directorySegment.header.id = segmentID; // 16 + directorySegment.header.allocatedSize = in.readLong(); // 8 + directorySegment.header.usedSize = in.readLong(); // 8 + directorySegment.data.entryCount = in.readInt(); // 4 -> Sum of bytes = 36 + in.skipBytes(124); // 128 - 4; + directorySegment.data.entries = new SubBlockDirectorySegment.SubBlockDirectorySegmentData.SubBlockDirectoryEntry[directorySegment.data.entryCount]; + for (int i=0; i Sum of all bytes so far = 36 + in.skipBytes(252); // 256 - 4; + directorySegment.data.entries = new AttachmentDirectorySegment.AttachmentDirectorySegmentData.AttachmentEntry[directorySegment.data.entryCount]; + for (int i=0; i probably useless because it we know the class, we know the schema type + public int pixelType; + public long filePosition; + public int filePart; + public int compression; + //public String _spare; + public int dimensionCount; + + // max. allocation for ease of use (valid size = 32 + EntryCount * 20) + //struct DimensionEntryDV DimensionEntries[MAXDIMENSIONS]; // offset 32 + public DimensionEntry[] dimensionEntries; // offset 32 + + public static class DimensionEntry { + /* + typedef struct PACKED DimensionEntry + { + char Dimension[4]; + std::int32_t Start; + std::int32_t Size; + float StartCoordinate; + std::int32_t StoredSize; + } DIMENSIONENTRY; + */ + + public String dimension; + public int start; + public int size; // real physical size + public float startCoordinate; // TODO : remove ? + public int storedSize; // number of pixels in this block + + @Override + public String toString() { + return "dimension=" + dimension + ", start=" + start + ", size=" + size + + ", startCoordinate=" + startCoordinate + ", storedSize=" + storedSize; + } + + + } + + @Override + public String toString() { + String s = "schemaType = DV, pixelType = " + pixelType + ", filePosition = " + + filePosition + ", filePart = " + filePart + ", compression = " + compression + + ", pyramidType = TODO" /*+ pyramidType*/ + ", dimensionCount = " + dimensionCount; + if (dimensionCount > 0) { + StringBuilder sb = new StringBuilder(s); + sb.append(", dimensions = ["); + for (int i=0; i AttachmentEntry entries[EntryCount]; + };*/ + public int entryCount; + public String _spare; // _spare[SIZE_ATTACHMENTDIRECTORY_DATA - 4]; + // followed by any sequence of SubBlockDirectoryEntryDE or SubBlockDirectoryEntryDV records; + public AttachmentEntry[] entries; + + public static class AttachmentEntry { + /* + struct PACKED AttachmentEntryA1 + { + unsigned char SchemaType[2]; + unsigned char _spare[10]; + std::int64_t FilePosition; + std::int32_t FilePart; + GUID ContentGuid; + unsigned char ContentFileType[8]; + unsigned char Name[80]; + }; + */ + + public String schemaType; + // Spare : 10 bytes + public long filePosition; + public int filePart; + // GUID : 16 bytes + public int compression; + public String contentFileType; + public String name; //dimensionCount; + + } + } + + } + + // --------------------------- CONSTANTS + + public final static String + ZISRAWFILE = "ZISRAWFILE", + ZISRAWDIRECTORY = "ZISRAWDIRECTORY", + ZISRAWATTDIR = "ZISRAWATTDIR", + ZISRAWMETADATA = "ZISRAWMETADATA", + ZISRAWATTACH = "ZISRAWATTACH"; + + // defined segment alignments (never modify this constants!) + final public static int SEGMENT_ALIGN = 32; + + // Sizes of segment parts (never modify this constants!) + final public static int SIZE_SEGMENTHEADER = 32; + final public static int SIZE_SEGMENTID = 16; + final public static int SIZE_SUBBLOCKDIRECTORYENTRY_DE = 128; + final public static int SIZE_ATTACHMENTENTRY = 128; + final public static int SIZE_SUBBLOCKDIRECTORYENTRY_DV_FIXEDPART = 32; + + // Data section within segments (never modify this constants!) + final public static int SIZE_FILEHEADER_DATA = 512; + final public static int SIZE_METADATA_DATA = 256; + final public static int SIZE_SUBBLOCKDATA_MINIMUM = 256; + final public static int SIZE_SUBBLOCKDATA_FIXEDPART = 16; + final public static int SIZE_SUBBLOCKDIRECTORY_DATA = 128; + final public static int SIZE_ATTACHMENTDIRECTORY_DATA = 256; + final public static int SIZE_ATTACHMENT_DATA = 256; + final public static int SIZE_DIMENSIONENTRYDV = 20; + + /** Pixel type constants. See CziUtils.cpp */ + public static final int GRAY8 = 0; + public static final int GRAY16 = 1; + public static final int GRAY_FLOAT = 2; + public static final int BGR_24 = 3; + public static final int BGR_48 = 4; + public static final int BGR_FLOAT = 8; + public static final int BGRA_8 = 9; + public static final int COMPLEX = 10; + public static final int COMPLEX_FLOAT = 11; + public static final int GRAY32 = 12; + public static final int GRAY_DOUBLE = 13; + + /** Compression constants. See CziUtils.cpp */ + public static final int UNCOMPRESSED = 0; + public static final int JPEG = 1; + public static final int LZW = 2; + public static final int JPEGXR = 4; + public static final int ZSTD_0 = 5; + public static final int ZSTD_1 = 6; + public static final int ALIGNMENT = 32; // all segments are aligned on 32 bytes increments + public static final int HEADER_SIZE = 32; // SubBlock header size + +} +/* + +///////////////////////////////////////////////////////////////////////////////// +// Enumerations +///////////////////////////////////////////////////////////////////////////////// + + + +//////////////////////////////////////////////////////////////////// +// STRUCTURES +//////////////////////////////////////////////////////////////////// + +typedef struct PACKED AttachmentInfo +{ + std::int64_t AllocatedSize; + std::int64_t DataSize; + std::int32_t FilePart; + GUID ContentGuid; + char ContentFileType[8]; + char Name[80]; + //HANDLE FileHandle; + unsigned char spare[128]; +} ATTACHMENTINFO; + +typedef struct PACKED MetadataInfo +{ + std::int64_t AllocatedSize; + std::int32_t XmlSize; + std::int32_t BinarySize; +} METADATAINFO; + +typedef struct PACKED AttachmentDirectoryInfo +{ + std::int32_t EntryCount; + //HANDLE* attachmentHandles; +} ATTACHMENTDIRECTORYINFO; + +//////////////////////////////////////////////////////////////////// +// COMMON +//////////////////////////////////////////////////////////////////// + +// internal implementation limits (internal use of pre-allocated structures) +// re-dimension if more items needed +const int MAXDIMENSIONS = 40; +//#define MAXFILE 50000 +// +//#define ATTACHMENT_SPARE 2048 + +//////////////////////////////////////////////////////////////////// +// SCHEMAS +//////////////////////////////////////////////////////////////////// + +// SubBlockDirectory - Entry: DV variable length - mimimum of 256 bytes + +/////////////////////////////////////////////////////////////////////////////////// +// Attachment + +struct PACKED AttachmentEntryA1 +{ + unsigned char SchemaType[2]; + unsigned char _spare[10]; + std::int64_t FilePosition; + std::int32_t FilePart; + GUID ContentGuid; + unsigned char ContentFileType[8]; + unsigned char Name[80]; +}; + +struct PACKED AttachmentSegmentData +{ + std::int64_t DataSize; + unsigned char _spare[8]; + union + { + std::uint8_t reserved[SIZE_ATTACHMENTENTRY]; + struct AttachmentEntryA1 entry; // offset 16 + }; + unsigned char _spare2[SIZE_ATTACHMENT_DATA - SIZE_ATTACHMENTENTRY - 16]; +}; + +struct PACKED AttachmentDirectorySegmentData +{ + std::int32_t EntryCount; + unsigned char _spare[SIZE_ATTACHMENTDIRECTORY_DATA - 4]; + // followed by => AttachmentEntry entries[EntryCount]; +}; + + +/////////////////////////////////////////////////////////////////////////////////// +// SubBlock + + + +/////////////////////////////////////////////////////////////////////////////////// +// Metadata + +struct PACKED MetadataSegmentData +{ + std::int32_t XmlSize; + std::int32_t AttachmentSize; + unsigned char _spare[SIZE_METADATA_DATA - 8]; +}; + + +//////////////////////////////////////////////////////////////////// +// SEGMENTS +//////////////////////////////////////////////////////////////////// + + + + + +// MetdataSegment: size = 128(fixed) + dataLength +struct PACKED MetadataSegment +{ + struct SegmentHeader header; + struct MetadataSegmentData data; +}; + +// AttachmentDirectorySegment: size = 256(fixed) + EntryCount * 128(fixed) +struct PACKED AttachmentDirectorySegment +{ + struct SegmentHeader header; + struct AttachmentDirectorySegmentData data; +}; + +// AttachmentSegment: size = 256(fixed) +struct PACKED AttachmentSegment +{ + struct SegmentHeader header; + struct AttachmentSegmentData data; +}; + + + */ From e6645e70d57550ad545350bb160c765eb3c84b40 Mon Sep 17 00:00:00 2001 From: David Gault Date: Thu, 16 Nov 2023 15:14:54 +0000 Subject: [PATCH 02/16] Fix initFile failures Small changes to fix some initFile failures --- components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java index 4f7088f8518..51c4e79f334 100644 --- a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java +++ b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java @@ -1267,7 +1267,7 @@ protected void initFile(String id) throws FormatException, IOException { HashMap> blocksInCore = mapCoreTZCToBlocks.get(iCoreIndex); HashMap> minimalBlocksInCore = coreIndexToTZCToMinimalBlocks.get(iCoreIndex); for (ModuloDimensionEntries block: coreSignatureToBlocks.get(coreSignature)) { - int c = block.getDimension("C").start; + int c = (block.hasDimension("C"))? block.getDimension("C").start: 0; int z = (block.hasDimension("Z"))? block.getDimension("Z").start: 0; int t = (block.hasDimension("T"))? block.getDimension("T").start: 0; CZTKey k = new CZTKey(c,z,t); From a2f1742c37cb4321585fd38a5d8f18c88aa61b70 Mon Sep 17 00:00:00 2001 From: David Gault Date: Thu, 16 Nov 2023 15:20:01 +0000 Subject: [PATCH 03/16] Fix initFile failure when unitsLength is reference frame zStep is always initialised as micrometer so no need to attempt unit conversion --- components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java index 51c4e79f334..6bcd4570b84 100644 --- a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java +++ b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java @@ -3439,7 +3439,7 @@ private void setSpaceAndTimeInformation( // of series and of planes offsetZ0 = planePosZ.value(unitLength).doubleValue(); } } - double stepZ = (zStep==null)?0:zStep.value(unitLength).doubleValue(); + double stepZ = (zStep==null)?0:zStep.value().doubleValue(); if (resolutionLevel0) timeStampsResolutionLevel0.put(iChannel, new double[reader.getSizeT()*reader.getSizeZ()]); // Store the timestamps to be copied over resolution levels double[] currentTimeStamps = timeStampsResolutionLevel0.get(iChannel); From 25686ef6258fa3f592e86e619306426a64e8ba38 Mon Sep 17 00:00:00 2001 From: Nicolas Chiaruttini Date: Sat, 16 Dec 2023 19:49:11 +0100 Subject: [PATCH 04/16] CZI reader: fixes incorrect optimisation checks that a single sublock exists when a cropped plane matches exactly a subblock region --- .../formats-gpl/src/loci/formats/in/ZeissCZIReader.java | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java index 6bcd4570b84..9921cbe954f 100644 --- a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java +++ b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java @@ -840,8 +840,8 @@ public byte[] openBytes(int no, byte[] buf, int x, int y, int w, int h) throws F if (image.intersects(blockRegion)) { RandomAccessInputStream stream = getStream(block.filePart); - if (image.equals(blockRegion)) { - // Best case scenario + if (image.equals(blockRegion) && blocks.size()==1) { // THE SECOND TEST IS NECESSARY BECAUSE OTHER BLOCKS CAN INTERSECT! + // Best case scenario: reads and returns full subblock return readRawPixelData( block, coreIndexToCompression.get(coreIndex), From c5b3c8f8855520132a259992230b68da659292a7 Mon Sep 17 00:00:00 2001 From: Nicolas Chiaruttini Date: Sat, 16 Dec 2023 20:14:02 +0100 Subject: [PATCH 05/16] Fix @JavascriptMick comments (partially) --- .../src/loci/formats/in/ZeissCZIReader.java | 13 +++---------- 1 file changed, 3 insertions(+), 10 deletions(-) diff --git a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java index 9921cbe954f..4fef9134741 100644 --- a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java +++ b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java @@ -421,16 +421,13 @@ protected ArrayList getAvailableOptions() { // -- ZeissCZI-specific methods -- - public static boolean allowAutoStitch = false; // TODO CHANGE THIS TO METADATA OPTIONS! - public boolean allowAutostitching() { - //return false; MetadataOptions options = getMetadataOptions(); if (options instanceof DynamicMetadataOptions) { return ((DynamicMetadataOptions) options).getBoolean( ALLOW_AUTOSTITCHING_KEY, ALLOW_AUTOSTITCHING_DEFAULT); } - return allowAutoStitch;// ALLOW_AUTOSTITCHING_DEFAULT;*/ + return ALLOW_AUTOSTITCHING_DEFAULT; } public boolean canReadAttachments() { // TODO : handle this method @@ -2286,13 +2283,9 @@ private void readXMLMetadata(LibCZI.MetaDataSegment metaDataSegment, DocumentBui private void translateMetadata(String xml, DocumentBuilder parser) throws FormatException, IOException { Element root; - try { - ByteArrayInputStream s = - new ByteArrayInputStream(xml.getBytes(Constants.ENCODING)); + try ( ByteArrayInputStream s = new ByteArrayInputStream(xml.getBytes(Constants.ENCODING))) { root = parser.parse(s).getDocumentElement(); - s.close(); - } - catch (SAXException e) { + } catch (SAXException e) { throw new FormatException(e); } From faed5a37d46759ffcc95bd06017240bedaaa0afc Mon Sep 17 00:00:00 2001 From: Nicolas Chiaruttini Date: Sun, 17 Dec 2023 18:07:44 +0100 Subject: [PATCH 06/16] Overrides getFillColor as in original CZI reader --- .../src/loci/formats/in/ZeissCZIReader.java | 27 ++++++++++++++++++- 1 file changed, 26 insertions(+), 1 deletion(-) diff --git a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java index 4fef9134741..c4059408ad6 100644 --- a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java +++ b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java @@ -297,7 +297,7 @@ public class ZeissCZIReader extends FormatReader { coreIndexToTZCToMinimalBlocks = new ArrayList<>(); @CopyByRef - int nIlluminations, nRotations, nPhases; + int nIlluminations, nRotations, nPhases, maxResolution; // ------------------------ METADATA FIELDS @CopyByRef @@ -470,6 +470,27 @@ public boolean isThisType(RandomAccessInputStream stream) throws IOException { return check.equals(CZI_MAGIC_STRING); } + /** + * @see loci.formats.FormatReader#getFillColor() + * + * If the fill value was set explicitly, use that. + * Otherwise, return 255 (white) for RGB data with a pyramid, + * and 0 in all other cases. RGB data with a pyramid can + * reasonably be assumed to be a brightfield slide. + */ + @Override + public Byte getFillColor() { + if (fillColor != null) { + return fillColor; + } + + byte fill = (byte) 0; + if (isRGB() && maxResolution > 0) { + fill = (byte) 255; + } + return fill; + } + private void swapRGBIfnecessary(byte[] buf, int compression, int bpp, int pixel) { if (isRGB() /*&& !emptyTile*/ && compression != JPEGXR) { // TODO: case emptytile // channels are stored in BGR order; red and blue channels need switching @@ -821,6 +842,8 @@ public byte[] openBytes(int no, byte[] buf, int x, int y, int w, int h) throws F baseResolution--; } + Arrays.fill(buf, getFillColor()); + // The data is somewhere in these blocks List blocks = coreIndexToTZCToMinimalBlocks.get(currentIndex).get(key); @@ -1051,6 +1074,8 @@ protected void initFile(String id) throws FormatException, IOException { nPhases = maxValuePerDimension.containsKey("H")? maxValuePerDimension.get("H")+1:1; + maxResolution = maxValuePerDimension.containsKey(RESOLUTION_LEVEL_DIMENSION)? maxValuePerDimension.get(RESOLUTION_LEVEL_DIMENSION):0; // Used only for auto-determination of the fill color + int nChannels = maxValuePerDimension.containsKey("C")? maxValuePerDimension.get("C")+1:1; int nSlices = maxValuePerDimension.containsKey("Z")? maxValuePerDimension.get("Z")+1:1; From f56381ee0ae66bbaeda1f7e94ae912aa51c93591 Mon Sep 17 00:00:00 2001 From: Nicolas Chiaruttini Date: Sun, 17 Dec 2023 18:16:44 +0100 Subject: [PATCH 07/16] Normalizes color as in PR https://github.com/ome/bioformats/pull/4088 --- .../src/loci/formats/in/ZeissCZIReader.java | 108 +++++++++++++++++- 1 file changed, 104 insertions(+), 4 deletions(-) diff --git a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java index c4059408ad6..cecf4d508cc 100644 --- a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java +++ b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java @@ -224,8 +224,6 @@ * maps the file 'temporarily' to a fake file. That's pretty clever and convenient, but prevents (most probably) * lazy loading AND memoization functionality. * - * TODO: sync https://github.com/ome/bioformats/pull/4088, that was fixed after this reader was branched from bio-formats - * TODO: implement getfillcolor * TODO: test PALM file * TODO: ask how to get rid of absolute file path in memo that do not crash the reader when the file is moved * @@ -284,9 +282,19 @@ public class ZeissCZIReader extends FormatReader { @CopyByRef private Map coreIndexToSeries = new HashMap<>(); + // This array has to be taken out of the metadata initializer because of the way IFormatReader#get8BitLookupTable + // and IFormatReader#get16BitLookupTable work + @CopyByRef + ArrayList channels = new ArrayList<>(); + // streamCurrentSeries is a temp field that should maybe be changed when setSeries is called transient int streamCurrentPart = -1; + // previous channel has a value set by the last bytes being called, this is a weird behaviour IMO + // but it behaves as expected to make the methods IFormatReader#get8BitLookupTable and + // IFormatReader#get16BitLookupTable work + transient int previousChannel = -1; + // Core map structure for fast access to blocks: // - first key: bio-formats core index // - second key: czt index @@ -470,6 +478,84 @@ public boolean isThisType(RandomAccessInputStream stream) throws IOException { return check.equals(CZI_MAGIC_STRING); } + /* @see loci.formats.IFormatReader#get8BitLookupTable() */ + @Override + public byte[][] get8BitLookupTable() throws FormatException, IOException { + if ((getPixelType() != FormatTools.INT8 && + getPixelType() != FormatTools.UINT8) || previousChannel == -1 || + previousChannel >= channels.size()) + { + return null; + } + + byte[][] lut = new byte[3][256]; + + String color = channels.get(previousChannel).color; + if (color != null) { + color = normalizeColor(color); + try { + int colorValue = Integer.parseInt(color, 16); + + int redMax = (colorValue & 0xff0000) >> 16; + int greenMax = (colorValue & 0xff00) >> 8; + int blueMax = colorValue & 0xff; + + for (int i=0; i= channels.size()) + { + return null; + } + + short[][] lut = new short[3][65536]; + + String color = channels.get(previousChannel).color; + if (color != null) { + color = normalizeColor(color); + try { + int colorValue = Integer.parseInt(color, 16); + + int redMax = (colorValue & 0xff0000) >> 16; + int greenMax = (colorValue & 0xff00) >> 8; + int blueMax = colorValue & 0xff; + + redMax = (int) (65535 * (redMax / 255.0)); + greenMax = (int) (65535 * (greenMax / 255.0)); + blueMax = (int) (65535 * (blueMax / 255.0)); + + for (int i=0; i 0 && @@ -1305,7 +1392,11 @@ protected void initFile(String id) throws FormatException, IOException { } // Initialize the reader store, and basically all metadata - new MetadataInitializer(this).initializeMetadata(cziPartToSegments, mapCoreTZCToBlocks); + MetadataInitializer mi = new MetadataInitializer(this); + mi.initializeMetadata(cziPartToSegments, mapCoreTZCToBlocks); + for (MetadataInitializer.Channel channel: mi.channels) { + channels.add(channel); + } } @@ -4228,7 +4319,7 @@ private void addChannelMetadata(int iSeries, boolean isPALM) { String color = channels.get(c).color; if (color != null && !reader.isRGB()) { - color = color.replaceAll("#", ""); + color = normalizeColor(color); if (color.length() > 6) { color = color.substring(2); } @@ -4641,4 +4732,13 @@ private boolean isLatticeLightSheet() { } + static private String normalizeColor(String color) { + String c = color.replaceAll("#", ""); + if (c.length() > 6) { + c = c.substring(2, (int) Math.min(8, c.length())); + LOGGER.debug("Replaced color {} with {}", color, c); + } + return c; + } + } From 0cbffad95873fb6edb3de0d23e48db5711942cb2 Mon Sep 17 00:00:00 2001 From: Nicolas Chiaruttini Date: Sun, 17 Dec 2023 18:18:26 +0100 Subject: [PATCH 08/16] Adds support of INCLUDE_ATTACHMENTS_KEY options --- .../src/loci/formats/in/ZeissCZIReader.java | 14 ++++++++------ 1 file changed, 8 insertions(+), 6 deletions(-) diff --git a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java index cecf4d508cc..0a8bdce0425 100644 --- a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java +++ b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java @@ -1327,12 +1327,14 @@ protected void initFile(String id) throws FormatException, IOException { List sortedFileParts = cziPartToSegments.keySet().stream().sorted().collect(Collectors.toList()); - try { - addLabelIfExists(sortedFileParts, cziPartToSegments, id);//, allPositionsInformation); - addSlidePreviewIfExists(sortedFileParts, cziPartToSegments, id);//, allPositionsInformation); - //getJPGThumbnailIfExists(sortedFileParts, cziPartToSegments, id); //disabled for bwd compatibility - } catch (DependencyException | ServiceException e) { - throw new RuntimeException(e); + if (canReadAttachments()) { + try { + addLabelIfExists(sortedFileParts, cziPartToSegments, id);//, allPositionsInformation); + addSlidePreviewIfExists(sortedFileParts, cziPartToSegments, id);//, allPositionsInformation); + //getJPGThumbnailIfExists(sortedFileParts, cziPartToSegments, id); //disabled for bwd compatibility + } catch (DependencyException | ServiceException e) { + throw new RuntimeException(e); + } } LOGGER.trace("#CoreSeries = {}", core.size()); From 01271eeec145c18daf5e7ba8cee727e8a9234c3c Mon Sep 17 00:00:00 2001 From: Nicolas Chiaruttini Date: Mon, 18 Dec 2023 08:02:04 +0100 Subject: [PATCH 09/16] Fix truncation issue in downsampling calculation Adds toString method to directory entry --- .../src/loci/formats/in/ZeissCZIReader.java | 8 ++++---- .../src/loci/formats/in/libczi/LibCZI.java | 15 +++++++++++++++ 2 files changed, 19 insertions(+), 4 deletions(-) diff --git a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java index 0a8bdce0425..1d509a38ab9 100644 --- a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java +++ b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java @@ -1161,8 +1161,6 @@ protected void initFile(String id) throws FormatException, IOException { nPhases = maxValuePerDimension.containsKey("H")? maxValuePerDimension.get("H")+1:1; - maxResolution = maxValuePerDimension.containsKey(RESOLUTION_LEVEL_DIMENSION)? maxValuePerDimension.get(RESOLUTION_LEVEL_DIMENSION):0; // Used only for auto-determination of the fill color - int nChannels = maxValuePerDimension.containsKey("C")? maxValuePerDimension.get("C")+1:1; int nSlices = maxValuePerDimension.containsKey("Z")? maxValuePerDimension.get("Z")+1:1; @@ -1198,7 +1196,7 @@ protected void initFile(String id) throws FormatException, IOException { cziPartToSegments.forEach((part, cziSegments) -> { // For each part Arrays.asList(cziSegments.subBlockDirectory.data.entries).forEach( // and each entry entry -> { - int downscalingFactor = entry.getDimension("X").size/entry.getDimension("X").storedSize; + int downscalingFactor = (int) Math.round((double)(entry.getDimension("X").size)/(double)(entry.getDimension("X").storedSize)); if ((downscalingFactor==1)||(allowAutostitching())) { // Split by resolution level if flattenedResolutions is true ModuloDimensionEntries moduloEntry = new ModuloDimensionEntries(entry, @@ -1219,6 +1217,8 @@ protected void initFile(String id) throws FormatException, IOException { }); }); + maxResolution = maxValuePerDimension.containsKey(RESOLUTION_LEVEL_DIMENSION)? maxValuePerDimension.get(RESOLUTION_LEVEL_DIMENSION):0; // Used only for auto-determination of the fill color + // Sort them List orderedCoreSignatureList = coreSignatureToBlocks.keySet().stream().sorted().collect(Collectors.toList()); @@ -1885,7 +1885,7 @@ public ModuloDimensionEntries(LibCZI.SubBlockDirectorySegment.SubBlockDirectoryS this.nPhases = nPhases; this.pixelType = entry.getPixelType(); this.compression = entry.getCompression(); - this.downSampling = entry.getDimension("X").size/entry.getDimension("X").storedSize; + this.downSampling = (int) Math.round((double)(entry.getDimension("X").size)/(double)(entry.getDimension("X").storedSize)); this.filePosition = entry.getFilePosition(); int iRotation = 0; diff --git a/components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java b/components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java index 3ba2d7e50fd..346b7b98329 100644 --- a/components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java +++ b/components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java @@ -631,6 +631,21 @@ public long getFilePosition() { return -1; } } + + @Override + public String toString() { + if (entryDV!=null) { + StringBuilder sb = new StringBuilder(); + sb.append("pixelType "+this.getPixelType()+" compression = "+getCompression()+"\n"); + for (SubBlockSegment.SubBlockSegmentData.SubBlockDirectoryEntryDV.DimensionEntry entry: getDimensionEntries()) { + sb.append(entry+"\n"); + } + return sb.toString(); + } else { + return "entryDE not supported"; + } + } + } } From 14c990c5329ca438acf85b5df0b2e389601b6ebb Mon Sep 17 00:00:00 2001 From: Nicolas Chiaruttini Date: Thu, 4 Jan 2024 16:41:44 +0100 Subject: [PATCH 10/16] Fix bug where rgb components are swapped while they shouldn't --- .../formats-gpl/src/loci/formats/in/ZeissCZIReader.java | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java index 1d509a38ab9..277ff9a1b42 100644 --- a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java +++ b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java @@ -776,7 +776,7 @@ private byte[] readRawPixelData(MinDimEntry block, if (buf != null && buf.length >= data.length) { System.arraycopy(data, 0, buf, 0, data.length); - swapRGBIfnecessary(buf, UNCOMPRESSED, bpp, totalBpp); + swapRGBIfnecessary(buf, compression, bpp, totalBpp); if (useCache) { cacheLock.lock(); // Block just computed @@ -790,7 +790,7 @@ private byte[] readRawPixelData(MinDimEntry block, } return buf; } - swapRGBIfnecessary(data, UNCOMPRESSED, bpp, totalBpp); + swapRGBIfnecessary(data, compression, bpp, totalBpp); if (useCache) { cacheLock.lock(); subBlockLRUCache.touch(block, data); From 37a73454021f699c7a3dfe155f2cdf048bff4982 Mon Sep 17 00:00:00 2001 From: Nicolas Chiaruttini Date: Thu, 4 Jan 2024 16:44:17 +0100 Subject: [PATCH 11/16] Fix RGB WSI images not having a white background --- .../src/loci/formats/in/ZeissCZIReader.java | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java index 277ff9a1b42..60ce59177a5 100644 --- a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java +++ b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java @@ -305,7 +305,10 @@ public class ZeissCZIReader extends FormatReader { coreIndexToTZCToMinimalBlocks = new ArrayList<>(); @CopyByRef - int nIlluminations, nRotations, nPhases, maxResolution; + int nIlluminations, nRotations, nPhases; + + @CopyByRef + boolean hasPyramid = false; // ------------------------ METADATA FIELDS @CopyByRef @@ -571,7 +574,7 @@ public Byte getFillColor() { } byte fill = (byte) 0; - if (isRGB() && maxResolution > 0) { + if (isRGB() && (hasPyramid)) { fill = (byte) 255; } return fill; @@ -1197,6 +1200,7 @@ protected void initFile(String id) throws FormatException, IOException { Arrays.asList(cziSegments.subBlockDirectory.data.entries).forEach( // and each entry entry -> { int downscalingFactor = (int) Math.round((double)(entry.getDimension("X").size)/(double)(entry.getDimension("X").storedSize)); + hasPyramid = hasPyramid || (downscalingFactor!=1); if ((downscalingFactor==1)||(allowAutostitching())) { // Split by resolution level if flattenedResolutions is true ModuloDimensionEntries moduloEntry = new ModuloDimensionEntries(entry, @@ -1217,8 +1221,6 @@ protected void initFile(String id) throws FormatException, IOException { }); }); - maxResolution = maxValuePerDimension.containsKey(RESOLUTION_LEVEL_DIMENSION)? maxValuePerDimension.get(RESOLUTION_LEVEL_DIMENSION):0; // Used only for auto-determination of the fill color - // Sort them List orderedCoreSignatureList = coreSignatureToBlocks.keySet().stream().sorted().collect(Collectors.toList()); From bab8d5a71290d4cc9bf6bc3c53a9aede67bd2a4c Mon Sep 17 00:00:00 2001 From: Nicolas Chiaruttini Date: Thu, 4 Jan 2024 16:53:17 +0100 Subject: [PATCH 12/16] Recovers file reading when subblock metadata has been deleted but is present as an orphan block at the end of the file. --- .../src/loci/formats/in/ZeissCZIReader.java | 10 ++- .../src/loci/formats/in/libczi/LibCZI.java | 73 ++++++++++++++++--- 2 files changed, 69 insertions(+), 14 deletions(-) diff --git a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java index 60ce59177a5..3af5efe0002 100644 --- a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java +++ b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java @@ -2108,9 +2108,13 @@ private static class CZISegments { public CZISegments(String id, boolean littleEndian) throws IOException { this.fileName = id; this.fileHeader = LibCZI.getFileHeaderSegment(id, BUFFER_SIZE, littleEndian); - this.subBlockDirectory = LibCZI.getSubBlockDirectorySegment(this.fileHeader, id, BUFFER_SIZE, littleEndian); - this.metadata = LibCZI.getMetaDataSegment(this.fileHeader, id, BUFFER_SIZE, littleEndian); - this.attachmentDirectory = LibCZI.getAttachmentDirectorySegment(this.fileHeader, id, BUFFER_SIZE, littleEndian); + this.subBlockDirectory = LibCZI.getSubBlockDirectorySegment(this.fileHeader.data.subBlockDirectoryPosition, id, BUFFER_SIZE, littleEndian); + this.attachmentDirectory = LibCZI.getAttachmentDirectorySegment(this.fileHeader.data.attachmentDirectoryPosition, id, BUFFER_SIZE, littleEndian); + + // For searching of blocks at the end of the file, in case the metadata subblock has been deleted + long lastBlockPosition = LibCZI.getPositionLastBlock(subBlockDirectory); + this.metadata = LibCZI.getMetaDataSegment(this.fileHeader.data.metadataPosition, id, BUFFER_SIZE, littleEndian, lastBlockPosition); + if (attachmentDirectory!=null) { this.timeStamps = LibCZI.getTimeStamps(this.attachmentDirectory, id, BUFFER_SIZE, littleEndian); //System.out.println("#ts="+timeStamps.length); diff --git a/components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java b/components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java index 346b7b98329..1203335cd26 100644 --- a/components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java +++ b/components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java @@ -83,20 +83,20 @@ public static FileHeaderSegment getFileHeaderSegment(String id, int BUFFER_SIZE, } /** - * @param fileHeader file header segment of the referenced czi file + * @param blockPosition position of the subblockdirectory segment of the referenced czi file * @param id a czi file path * @param BUFFER_SIZE the size of the caching buffer in bytes * @param isLittleEndian endianness of the data * @return the subblock directory segment of the czi file * @throws IOException invalid file, invalid file location, segment not found */ - public static SubBlockDirectorySegment getSubBlockDirectorySegment(FileHeaderSegment fileHeader, String id, int BUFFER_SIZE, boolean isLittleEndian) throws IOException { + public static SubBlockDirectorySegment getSubBlockDirectorySegment(long blockPosition, String id, int BUFFER_SIZE, boolean isLittleEndian) throws IOException { // TODO (maybe) : increase buffer size to limit the number of IO calls, especially when the file is mounted on a network drive try (RandomAccessInputStream in = new RandomAccessInputStream(id, BUFFER_SIZE)) { SubBlockDirectorySegment directorySegment = new SubBlockDirectorySegment(); in.order(isLittleEndian); - in.seek(fileHeader.data.subBlockDirectoryPosition); + in.seek(blockPosition); String segmentID = in.readString(16).trim(); if (segmentID.equals(ZISRAWDIRECTORY)) { @@ -126,18 +126,18 @@ public static SubBlockDirectorySegment getSubBlockDirectorySegment(FileHeaderSeg } /** - * @param fileHeader file header segment of the referenced czi file + * @param blockPosition position of the attachment directory segment of the referenced czi file * @param id a czi file path * @param BUFFER_SIZE the size of the caching buffer in bytes * @param isLittleEndian endianness of the data * @return the attachment directory segment * @throws IOException invalid file, invalid file location, segment not found */ - public static AttachmentDirectorySegment getAttachmentDirectorySegment(FileHeaderSegment fileHeader, String id, int BUFFER_SIZE, boolean isLittleEndian) throws IOException { + public static AttachmentDirectorySegment getAttachmentDirectorySegment(long blockPosition, String id, int BUFFER_SIZE, boolean isLittleEndian) throws IOException { try (RandomAccessInputStream in = new RandomAccessInputStream(id, BUFFER_SIZE)) { AttachmentDirectorySegment directorySegment = new AttachmentDirectorySegment(); in.order(isLittleEndian); - in.seek(fileHeader.data.attachmentDirectoryPosition); + in.seek(blockPosition); String segmentID = in.readString(16).trim(); if (segmentID.equals(ZISRAWATTDIR)) { directorySegment.header.id = segmentID; // 16 bytes @@ -165,19 +165,35 @@ public static AttachmentDirectorySegment getAttachmentDirectorySegment(FileHeade } /** - * @param fileHeader file header segment of the referenced czi file + * @param blockPosition position of the expected MetaData Segment * @param id a czi file path * @param BUFFER_SIZE the size of the caching buffer in bytes * @param isLittleEndian endianness of the data * @return the raw metadata segment * @throws IOException invalid file, invalid file location, segment not found */ - public static MetaDataSegment getMetaDataSegment(FileHeaderSegment fileHeader, String id, int BUFFER_SIZE, boolean isLittleEndian) throws IOException { + public static MetaDataSegment getMetaDataSegment(long blockPosition, String id, int BUFFER_SIZE, boolean isLittleEndian, long lastBlockPosition) throws IOException { try (RandomAccessInputStream in = new RandomAccessInputStream(id, BUFFER_SIZE)) { in.order(isLittleEndian); - in.seek(fileHeader.data.metadataPosition); + in.seek(blockPosition); String segmentID = in.readString(16).trim(); - if (segmentID.equals(ZISRAWMETADATA)) { + boolean metadataSegmentFound = false; + + // ------------ Handling of potentially deleted metadata segement + if (!segmentID.equals(ZISRAWMETADATA)) { + if (segmentID.equals(DELETED)) { + // The metadata segment has been deleted, let's walk at the end of the file and attempt to find the missing metadata segment + long newPosition = LibCZI.findOrphanBlock(id, BUFFER_SIZE, isLittleEndian, lastBlockPosition, ZISRAWMETADATA); + if (newPosition != -1) { + return getMetaDataSegment(newPosition, id, BUFFER_SIZE, isLittleEndian, -1); + } else { + throw new IOException(ZISRAWMETADATA+" segment expected, found "+segmentID+" instead."); + } + } + } else { + metadataSegmentFound = true; + } + if (metadataSegmentFound) { MetaDataSegment metaDataSegment = new MetaDataSegment(); // read the segment header metaDataSegment.header.id = segmentID; @@ -512,6 +528,40 @@ public static SubBlockMeta readSubBlockMeta(RandomAccessInputStream in, SubBlock return subBlockMeta; } + // If blocks have been deleted, they are appended at the end of the file. This method looks for the location of the last known block and + // iterates over the next blocks until the end of the file. This method returns whether some orphan blocks have been found + static long findOrphanBlock(String id, int BUFFER_SIZE, boolean isLittleEndian, long positionOfLastKnownBlock, String blockType) throws IOException { + try (RandomAccessInputStream in = new RandomAccessInputStream(id, BUFFER_SIZE)) { + long eof = in.length(); + in.order(isLittleEndian); + long positionBlockStart = positionOfLastKnownBlock; + in.seek(positionBlockStart); + String segmentID = in.readString(16).trim(); + + long allocatedSize = in.readLong(); + long usedSize = in.readLong(); + while (positionBlockStart + allocatedSize + 32 Date: Sun, 4 Feb 2024 20:02:37 +0100 Subject: [PATCH 13/16] Fis issue with CZI edge cases: - Fix mismatch between decompressed JPEGXR image dimension and subblock storedSize X & Y - Proper reading of linescans (handling of subblocks containing multiple Z or Ts) - Fix unit issue with linescans - Avoids NPE issue when no scaling metadata is found - adapts to PALM: make sure the data opens (downscaling can go below 1: superresolution) - Make metadata reading robust with absence of some sub-blocks - Correctly counts the number of channels, slices, and timepoints (they should always start at 0) - be robust to absence of sub-directory segment - There may be some information in slide previews that we do not want to keep. Hence a DummyMetadata is used when reading slide preview and macro images Minor documentation and formatting changes --- .../src/loci/formats/in/ZeissCZIReader.java | 479 +++++++++++------- .../src/loci/formats/in/libczi/LibCZI.java | 103 ++-- 2 files changed, 349 insertions(+), 233 deletions(-) diff --git a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java index 3af5efe0002..7e6b43a9ad8 100644 --- a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java +++ b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java @@ -9,15 +9,15 @@ * %% * 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 2 of the + * published by the Free Software Foundation, either version 2 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 + * + * You should have received a copy of the GNU General Public * License along with this program. If not, see * . * #L% @@ -58,7 +58,6 @@ import loci.common.Region; import loci.common.services.DependencyException; import loci.common.services.ServiceException; -import loci.common.services.ServiceFactory; import loci.common.xml.XMLTools; import loci.formats.CoreMetadata; import loci.formats.FormatException; @@ -70,13 +69,11 @@ import loci.formats.codec.JPEGXRCodec; import loci.formats.codec.LZWCodec; import loci.formats.codec.ZstdCodec; -import loci.formats.in.DynamicMetadataOptions; -import loci.formats.in.JPEGReader; -import loci.formats.in.MetadataOptions; import loci.formats.in.libczi.LibCZI; +import loci.formats.meta.DummyMetadata; import loci.formats.meta.MetadataStore; -import loci.formats.ome.OMEXMLMetadata; -import loci.formats.services.OMEXMLService; +import loci.formats.tiff.IFD; +import loci.formats.tiff.TiffParser; import ome.units.UNITS; import ome.units.quantity.Length; import ome.units.quantity.Power; @@ -223,8 +220,7 @@ * but loading these bytes on demand is quite tedious: hard to explain, but a reader is created inside the reader and * maps the file 'temporarily' to a fake file. That's pretty clever and convenient, but prevents (most probably) * lazy loading AND memoization functionality. - * - * TODO: test PALM file + *

* TODO: ask how to get rid of absolute file path in memo that do not crash the reader when the file is moved * */ @@ -293,7 +289,7 @@ public class ZeissCZIReader extends FormatReader { // previous channel has a value set by the last bytes being called, this is a weird behaviour IMO // but it behaves as expected to make the methods IFormatReader#get8BitLookupTable and // IFormatReader#get16BitLookupTable work - transient int previousChannel = -1; + transient int previousChannel = 0; // Core map structure for fast access to blocks: // - first key: bio-formats core index @@ -354,15 +350,20 @@ public ZeissCZIReader() { } /** Duplicates 'that' reader for parallel reading. - * Creating a reader with this constructor allows to keep a very low memory footprint + * Creating another reader using this constructor allows to keep a very low memory footprint * because all immutable objects are re-used by reference. * WARNING: calling {@link ZeissCZIReader#close()} on this or that reader will prevent the use - * of the other reader created with this constructor */ + * of the other reader created with this constructor + * WARNING: 'that' reader should have been initialized with setId before creating another reader + * */ public ZeissCZIReader(ZeissCZIReader that) { super(FORMAT, SUFFIX); domains = new String[] {FormatTools.LM_DOMAIN, FormatTools.HISTOLOGY_DOMAIN}; suffixSufficient = false; suffixNecessary = false; + if ((that.currentId ==null)||(that.currentId == "")) { + throw new RuntimeException("Do not duplicate this reader from a model if the model has not been initialized"); + } this.streamCurrentPart = -1; @@ -441,7 +442,7 @@ public boolean allowAutostitching() { return ALLOW_AUTOSTITCHING_DEFAULT; } - public boolean canReadAttachments() { // TODO : handle this method + public boolean canReadAttachments() { MetadataOptions options = getMetadataOptions(); if (options instanceof DynamicMetadataOptions) { return ((DynamicMetadataOptions) options).getBoolean( @@ -559,29 +560,17 @@ public short[][] get16BitLookupTable() throws FormatException, IOException { else return null; } - /** - * @see loci.formats.FormatReader#getFillColor() - * - * If the fill value was set explicitly, use that. - * Otherwise, return 255 (white) for RGB data with a pyramid, - * and 0 in all other cases. RGB data with a pyramid can - * reasonably be assumed to be a brightfield slide. - */ - @Override - public Byte getFillColor() { - if (fillColor != null) { - return fillColor; - } - - byte fill = (byte) 0; - if (isRGB() && (hasPyramid)) { - fill = (byte) 255; + static private String normalizeColor(String color) { + String c = color.replaceAll("#", ""); + if (c.length() > 6) { + c = c.substring(2, Math.min(8, c.length())); + LOGGER.debug("Replaced color {} with {}", color, c); } - return fill; + return c; } private void swapRGBIfnecessary(byte[] buf, int compression, int bpp, int pixel) { - if (isRGB() /*&& !emptyTile*/ && compression != JPEGXR) { // TODO: case emptytile + if (isRGB() && compression != JPEGXR) { // channels are stored in BGR order; red and blue channels need switching // JPEG-XR data has already been reversed during decompression int redOffset = bpp * 2; @@ -601,13 +590,25 @@ private void swapRGBIfnecessary(byte[] buf, int compression, int bpp, int pixel) } } - private byte[] readRawPixelData(MinDimEntry block, - int compression, - int storedSizeX, - int storedSizeY, - RandomAccessInputStream s, Region tile, byte[] buf, - int bpp, int totalBpp) throws FormatException, IOException { - //s.order(isLittleEndian()); -> it is already set when calling the method + private byte[] readRawPixelData( + CZTKey key, + MinDimEntry block, + int compression, + int storedSizeX, + int storedSizeY, + RandomAccessInputStream s, Region tile, byte[] buf, + int bpp, int totalBpp) throws FormatException, IOException { + //s.order(isLittleEndian()); -> unnecessary because it is already set when calling the method + + if ((key.t!=block.dimensionStartT)||(key.z!=block.dimensionStartZ)) { + // This code branch is used with some czi files: + // - line scans + // - with subblocks that may contain with multiple Ts or Zs + if (compression!=UNCOMPRESSED) { + logger.error("Compression is not supported with line scans."); + return null; + } + } if ((useCache)&&(compression!=UNCOMPRESSED)) { cacheLock.lock(); @@ -664,6 +665,12 @@ private byte[] readRawPixelData(MinDimEntry block, long blockDataOffset = subBlock.dataOffset; long blockDataSize = subBlock.data.dataSize; + if ((key.t!=block.dimensionStartT)||(key.z!=block.dimensionStartZ)) { + // Line scan with multiple T or Z per subblock + assert compression==UNCOMPRESSED; + blockDataOffset+=((key.t-block.dimensionStartT)+(key.z-block.dimensionStartZ))*totalBpp*Math.max(storedSizeX,storedSizeY); + } + s.seek(blockDataOffset); if (compression == UNCOMPRESSED) { @@ -688,7 +695,7 @@ private byte[] readRawPixelData(MinDimEntry block, options.interleaved = isInterleaved(); options.littleEndian = isLittleEndian(); options.bitsPerSample = bytesPerPixel * 8; - options.maxBytes = getSizeX() * getSizeY() * getRGBChannelCount() * bytesPerPixel; + options.maxBytes = block.storedSizeX * block.storedSizeY * getRGBChannelCount() * bytesPerPixel; // The maximal size is the one of the subblock switch (compression) { case JPEG: @@ -703,7 +710,11 @@ private byte[] readRawPixelData(MinDimEntry block, options.maxBytes = options.width * options.height * getRGBChannelCount() * bytesPerPixel; try { - data = new JPEGXRCodec().decompress(data, options); + byte[] decompressed = new JPEGXRCodec().decompress(data, options); + data = fixUnexpectedJPEGXRDimensions(data, decompressed, + block.storedSizeX, + block.storedSizeY, + totalBpp, getFillColor()); } catch (FormatException e) { if (data.length == options.maxBytes) { @@ -806,6 +817,62 @@ private byte[] readRawPixelData(MinDimEntry block, return data; } + private static byte[] fixUnexpectedJPEGXRDimensions(byte[] compressed, byte[] uncompressed, + int storedSizeX, int storedSizeY, + int totalBpp, Byte fillColor) { + // Sometimes (see post https://forum.image.sc/t/would-anyone-have-a-palm-czi-example-file/85900/12), + // decompressed subblock does not return the number of pixels expected from subblock field storedSizeX or storedSizeY + // to find an example that will use this correction, execute the code below + // on the Young-Mouse czi image from - resolution level 6 https://zenodo.org/records/10577621: + /* ZeissQuickStartCZIReader r = new ZeissQuickStartCZIReader(); + r.setId("image path to \\Young_mouse.czi"); + r.setSeries(5); + r.openPlane(0,0,0,5947,2168); */ + int expectedRawDataSize = storedSizeX*storedSizeY*totalBpp; + if (uncompressed.length!=expectedRawDataSize) { + // We got an issue + try { + int[] result = getWidthAndHeightFromJPEGXRBytes(compressed); // It is possible to get the dimension of the decompressed subblock + // thanks to a header present in the compressed bytes (check JPEGXR specs in the getWidthAndHeight method) + int w = result[0]; + int h = result[1]; + if ((w>storedSizeX)||(h>storedSizeY)) { // No handling of a decompressed subblock bigger than expected. Only smaller. + throw new RuntimeException("Too many pixels found in a CZI JPEGXR compressed subblock."); + } + assert uncompressed.length == (w * h * totalBpp); + byte[] corrected = new byte[expectedRawDataSize]; + Arrays.fill(corrected, fillColor); + // The strategy is to copy one line after another, leaving empty columns or line depending on + // the case. The color of the empty pixels is set by the fillColor argument + for (int y = 0; y < h; y++) { + System.arraycopy(uncompressed, w*totalBpp*y, corrected, storedSizeX*totalBpp*y, w*totalBpp); + } + return corrected; + } catch (FormatException e) { + throw new RuntimeException(e); + } catch (IOException e) { + throw new RuntimeException(e); + } + } else { + return uncompressed; + } + } + + // see table A.4 in ITU-T T.832 https://www.itu.int/rec/T-REC-T.832/en version 2009 + private static final int IMAGE_WIDTH_TAG = 0xBC80; + private static final int IMAGE_HEIGHT_TAG = 0xBC81; + + private static int[] getWidthAndHeightFromJPEGXRBytes(byte[] stream) throws FormatException, IOException { + try (RandomAccessInputStream s = new RandomAccessInputStream(stream)) { + s.order(true); + s.seek(4); + long ifdPointer = s.readInt(); + TiffParser p = new TiffParser(s); + IFD ifd = p.getIFD(ifdPointer); + return new int[]{ifd.getIFDIntValue(IMAGE_WIDTH_TAG), ifd.getIFDIntValue(IMAGE_HEIGHT_TAG)}; + } + } + private static int readVarint(RandomAccessInputStream stream) throws IOException { byte a = stream.readByte(); // if high bit set, read next byte @@ -894,6 +961,26 @@ public int getOptimalTileHeight() { } } + /** + * @see loci.formats.FormatReader#getFillColor() + * + * If the fill value was set explicitly, use that. + * Otherwise, return 255 (white) for RGB data with a pyramid, + * and 0 in all other cases. RGB data with a pyramid can + * reasonably be assumed to be a brightfield slide. + */ + @Override + public Byte getFillColor() { + if (fillColor != null) { + return fillColor; + } + byte fill = (byte) 0; + if (isRGB() && (hasPyramid)) { + fill = (byte) 255; + } + return fill; + } + @Override public byte[] openBytes(int no, byte[] buf, int x, int y, int w, int h) throws FormatException, IOException { @@ -953,6 +1040,7 @@ public byte[] openBytes(int no, byte[] buf, int x, int y, int w, int h) throws F if (image.equals(blockRegion) && blocks.size()==1) { // THE SECOND TEST IS NECESSARY BECAUSE OTHER BLOCKS CAN INTERSECT! // Best case scenario: reads and returns full subblock return readRawPixelData( + key, block, coreIndexToCompression.get(coreIndex), block.storedSizeX, @@ -975,6 +1063,7 @@ public byte[] openBytes(int no, byte[] buf, int x, int y, int w, int h) throws F Region tileInBlock = new Region(regionRead.x-blockRegion.x, regionRead.y-blockRegion.y, regionRead.width, regionRead.height); byte[] rawData = readRawPixelData( + key, block, compression, block.storedSizeX, @@ -983,7 +1072,7 @@ public byte[] openBytes(int no, byte[] buf, int x, int y, int w, int h) throws F compression==UNCOMPRESSED? DataTools.allocate(tileInBlock.width, tileInBlock.height, nCh, bpp): null, bpp, bytesPerPixel); - // We need to basically crop a rectangle with a rectangle, of potentially different sizes + // We need to crop a rectangle with another rectangle, of potentially different sizes // Let's find out the position of the block in the image referential int blockOriX = regionRead.x-image.x; int skipBytesStartX = 0; @@ -1015,7 +1104,7 @@ public byte[] openBytes(int no, byte[] buf, int x, int y, int w, int h) throws F int offsetRawData = skipLinesRawDataStart*nBytesPerLineRawData+skipBytesStartX; int offsetBuf = skipLinesBufStart*nBytesPerLineBuf+skipBytesBufStartX; - for (int i=0; i { // For each part - Arrays.asList(cziSegments.subBlockDirectory.data.entries).forEach( // and each entry - entry -> { - for (LibCZI.SubBlockSegment.SubBlockSegmentData.SubBlockDirectoryEntryDV.DimensionEntry dimEntry: entry.getDimensionEntries()) { - //int nDigits = String.valueOf(dimEntry.start).length(); // TODO: Can this be negative ? - int val = dimEntry.start; - if (!maxValuePerDimension.containsKey(dimEntry.dimension)) { - maxValuePerDimension.put(dimEntry.dimension, dimEntry.start); - } else { - int curMax = maxValuePerDimension.get(dimEntry.dimension); - if (val>curMax) { - maxValuePerDimension.put(dimEntry.dimension, val); + if (cziSegments.subBlockDirectory!=null) { + Arrays.asList(cziSegments.subBlockDirectory.data.entries).forEach( // and each entry + entry -> { + for (LibCZI.SubBlockSegment.SubBlockSegmentData.SubBlockDirectoryEntryDV.DimensionEntry dimEntry : entry.getDimensionEntries()) { + //int nDigits = String.valueOf(dimEntry.start).length(); // TODO: Can this be negative ? + int val = dimEntry.start; + if (!maxValuePerDimension.containsKey(dimEntry.dimension)) { + maxValuePerDimension.put(dimEntry.dimension, dimEntry.start); + } else { + int curMax = maxValuePerDimension.get(dimEntry.dimension); + if (val>curMax) { + maxValuePerDimension.put(dimEntry.dimension, val); + } } } } - } - ); + ); + } }); nIlluminations = maxValuePerDimension.containsKey("I")? maxValuePerDimension.get("I")+1:1; @@ -1192,38 +1285,47 @@ protected void initFile(String id) throws FormatException, IOException { // Ready to build the signature Map> coreSignatureToBlocks = new HashMap<>(); - maxDigitPerDimension.put(RESOLUTION_LEVEL_DIMENSION,5); // Let's hope that the downsampling ratio never exceeds 9999 TODO : improve + maxDigitPerDimension.put(RESOLUTION_LEVEL_DIMENSION,5); // It is assumed that the downsampling ratio never exceeds 99999 maxDigitPerDimension.put(FILE_PART_DIMENSION, String.valueOf(cziPartToSegments.size()).length()); // Write all signatures cziPartToSegments.forEach((part, cziSegments) -> { // For each part - Arrays.asList(cziSegments.subBlockDirectory.data.entries).forEach( // and each entry - entry -> { - int downscalingFactor = (int) Math.round((double)(entry.getDimension("X").size)/(double)(entry.getDimension("X").storedSize)); - hasPyramid = hasPyramid || (downscalingFactor!=1); - if ((downscalingFactor==1)||(allowAutostitching())) { - // Split by resolution level if flattenedResolutions is true - ModuloDimensionEntries moduloEntry = new ModuloDimensionEntries(entry, - nRotations, nIlluminations, nPhases, - nChannels, nSlices, nFrames, part); - - CoreSignature coreSignature = new CoreSignature(moduloEntry - , RESOLUTION_LEVEL_DIMENSION, - downscalingFactor,//getDownSampling(entry), - maxDigitPerDimension::get, - allowAutostitching(), - FILE_PART_DIMENSION, part); - if (!coreSignatureToBlocks.containsKey(coreSignature)) { - coreSignatureToBlocks.put(coreSignature, new ArrayList<>()); + if (cziSegments.subBlockDirectory!=null) { + Arrays.asList(cziSegments.subBlockDirectory.data.entries).forEach( // and each entry + entry -> { + double doubleDownscalingFactor = (double)(entry.getDimension("X").size) / (double)(entry.getDimension("X").storedSize); + if (doubleDownscalingFactor<1) { + // PALM dataset -> forcing pyramid level to 0 will lead to create a new series + doubleDownscalingFactor = 0; } - coreSignatureToBlocks.get(coreSignature).add(moduloEntry); - } - }); + int downscalingFactor = (int) Math.round(doubleDownscalingFactor); + + hasPyramid = hasPyramid || (downscalingFactor>1); + + if ((downscalingFactor==1)||(allowAutostitching())) { + // Split by resolution level if flattenedResolutions is true + ModuloDimensionEntries moduloEntry = new ModuloDimensionEntries(entry, + nRotations, nIlluminations, nPhases, + nChannels, nSlices, nFrames, part); + + CoreSignature coreSignature = new CoreSignature(moduloEntry + , RESOLUTION_LEVEL_DIMENSION, + downscalingFactor,//getDownSampling(entry), + maxDigitPerDimension::get, + allowAutostitching(), + FILE_PART_DIMENSION, part); + if (!coreSignatureToBlocks.containsKey(coreSignature)) { + coreSignatureToBlocks.put(coreSignature, new ArrayList<>()); + } + coreSignatureToBlocks.get(coreSignature).add(moduloEntry); + } + }); + } }); // Sort them List orderedCoreSignatureList = coreSignatureToBlocks.keySet().stream().sorted().collect(Collectors.toList()); - + // orderedCoreSignatureList.forEach(System.out::println); // We now know how many core index are present in the image... except for extra images! core = new ArrayList<>(); @@ -1322,17 +1424,16 @@ protected void initFile(String id) throws FormatException, IOException { coreIndexToSignature.add(coreSignature); } - // Add extra images / thumbnails: - // Label - // SlidePreview - // JPG Thumbnail -> not read for the sake of backward compatibility - List sortedFileParts = cziPartToSegments.keySet().stream().sorted().collect(Collectors.toList()); if (canReadAttachments()) { + // Add extra images / thumbnails: + // Label + // SlidePreview + // JPG Thumbnail -> not read for the sake of backward compatibility try { - addLabelIfExists(sortedFileParts, cziPartToSegments, id);//, allPositionsInformation); - addSlidePreviewIfExists(sortedFileParts, cziPartToSegments, id);//, allPositionsInformation); + addLabelIfExists(sortedFileParts, cziPartToSegments, id); + addSlidePreviewIfExists(sortedFileParts, cziPartToSegments, id); //getJPGThumbnailIfExists(sortedFileParts, cziPartToSegments, id); //disabled for bwd compatibility } catch (DependencyException | ServiceException e) { throw new RuntimeException(e); @@ -1391,6 +1492,41 @@ protected void initFile(String id) throws FormatException, IOException { MinDimEntry mde = new MinDimEntry(block); // Makes a trimmed down version of the block in order to reduce the reader memory footprint getMinimalEntry(block);// blocksInCore.get(k).add(mde); minimalBlocksInCore.get(k).add(mde); + + // Need to handle LineScans: there may be multiple Z or T per subblock + // We add the same MinDimEntry with all keys that need to access the same block + // We can retrieve the z or t offset thanks to the difference between the key + // and the startdimension. The offset is taken into account into the readrawdata method + if (block.hasDimension("Z")) { + int sizeZ = block.getDimension("Z").storedSize; + if (sizeZ>1) { + logger.debug("Multiple Z found in a single sub-block."); + for (int zi = z+1; zi < z+sizeZ; zi++) { + k = new CZTKey(c,zi,t); + if (!minimalBlocksInCore.containsKey(k)) { + blocksInCore.put(k, new ArrayList<>()); + minimalBlocksInCore.put(k, new ArrayList<>()); + } + blocksInCore.get(k).add(mde); + minimalBlocksInCore.get(k).add(mde); + } + } + } + if (block.hasDimension("T")) { + int sizeT = block.getDimension("T").storedSize; + if (sizeT>1) { + logger.debug("Multiple T found in a single sub-block."); + for (int ti = t+1; ti < t+sizeT; ti++) { + k = new CZTKey(c,z,ti); + if (!minimalBlocksInCore.containsKey(k)) { + blocksInCore.put(k, new ArrayList<>()); + minimalBlocksInCore.put(k, new ArrayList<>()); + } + blocksInCore.get(k).add(mde); + minimalBlocksInCore.get(k).add(mde); + } + } + } } //In the end, there are 'blocksInCore.values().size()' blocks in the core 'iCoreIndex' } @@ -1401,7 +1537,6 @@ protected void initFile(String id) throws FormatException, IOException { for (MetadataInitializer.Channel channel: mi.channels) { channels.add(channel); } - } private void addLabelIfExists(List sortedFileParts, Map cziPartToSegments, String id) throws IOException, FormatException, DependencyException, ServiceException {//}, AllPositionsInformation allPositionsInformation) throws IOException, FormatException, DependencyException, ServiceException { @@ -1409,15 +1544,12 @@ private void addLabelIfExists(List sortedFileParts, Map sortedFileParts, Mapt_min) minT = t_min; } - ms0.sizeZ = maxZ - minZ; - ms0.sizeC = maxC - minC; - ms0.sizeT = maxT - minT; + if (minZ!=0) logger.warn("No block found with Z = 0, first Z block found at Z = "+minZ); + if (minC!=0) logger.warn("No block found with C = 0, first C block found at C = "+minC); + if (minT!=0) logger.warn("No block found with T = 0, first T block found at T = "+minT); + + ms0.sizeZ = maxZ - 0;//minZ; + ms0.sizeC = maxC - 0;//minC; + ms0.sizeT = maxT - 0;//minT; if ((downScale!=1)&&(allowAutostitching())) { ms0.sizeX = nPixX_maxRes/downScale; @@ -1666,7 +1799,7 @@ private static void convertPixelType(CoreMetadata ms0, int pixelType) throws For * The M dimension 'mosaic' will split subblocks between cores only if autostitch is false. * * Actually, transforming the reader to allow it to merge all scenes together is as simple as - * returning true for the dimension "S" (and "M") + * returning true for the dimension "S" */ private static boolean ignoreDimForSeries(String dimension, boolean autostitch) { switch (dimension) { @@ -1676,7 +1809,6 @@ private static boolean ignoreDimForSeries(String dimension, boolean autostitch) case "T": case "C": case FILE_PART_DIMENSION: - //case "S": return true; case "M": return autostitch; @@ -1751,7 +1883,7 @@ static class CoreSignature implements Comparable { public int getFilePart() { return filePart; } - public CoreSignature(ModuloDimensionEntries entries, //LibCZI.SubBlockSegment.SubBlockSegmentData.SubBlockDirectoryEntryDV.DimensionEntry[] entries, + public CoreSignature(ModuloDimensionEntries entries, String pyramidLevelDimension, int pyramidLevelValue, Function maxDigitPerDimension, boolean autostitch, String filePartDimension, int filePartValue) { @@ -1875,7 +2007,6 @@ static class ModuloDimensionEntries { ms0.moduloT.type = FormatTools.PHASE; ms0.sizeT *= phases; */ - final List entryList = new ArrayList<>(); final int nRotations, nIlluminations, nPhases, filePart; @@ -1887,7 +2018,8 @@ public ModuloDimensionEntries(LibCZI.SubBlockDirectorySegment.SubBlockDirectoryS this.nPhases = nPhases; this.pixelType = entry.getPixelType(); this.compression = entry.getCompression(); - this.downSampling = (int) Math.round((double)(entry.getDimension("X").size)/(double)(entry.getDimension("X").storedSize)); + int ds = (int) Math.round((double)(entry.getDimension("X").size)/(double)(entry.getDimension("X").storedSize)); + this.downSampling = Math.max(ds,1); // The downsampling factor could go below 1 for PALM dataset, but within the block, the real data is such that there's no downsampling - just the pixel size changes this.filePosition = entry.getFilePosition(); int iRotation = 0; @@ -2023,17 +2155,18 @@ public String toString() { } } - - /** * A stripped down version of * {@link LibCZI.SubBlockDirectorySegment.SubBlockDirectorySegmentData.SubBlockDirectoryEntry} * Because we have really many of these objects, and it's critical to keep these objects as small as possible */ static class MinDimEntry { - + final long filePosition; + final int dimensionStartX, dimensionStartY; + final int storedSizeX, storedSizeY; final int dimensionStartZ; - + final int dimensionStartT; + final int filePart; public MinDimEntry(ModuloDimensionEntries entry) { filePosition = entry.getFilePosition(); dimensionStartX = entry.getDimension("X").start; @@ -2043,40 +2176,18 @@ public MinDimEntry(ModuloDimensionEntries entry) { } else { dimensionStartZ = 0; } + if (entry.hasDimension("T")) { + dimensionStartT = entry.getDimension("T").start; + } else { + dimensionStartT = 0; + } storedSizeX = entry.getDimension("X").storedSize; storedSizeY = entry.getDimension("Y").storedSize; filePart = entry.filePart; } - final long filePosition; - final int dimensionStartX, dimensionStartY; - final int storedSizeX, storedSizeY; - - final int filePart; - - } - - //MinimalDimensionEntry - for Java > 17 - /*public record MinDimEntry(int dimensionStartZ, long filePosition, - - int dimensionStartX,int dimensionStartY, int storedSizeX, int storedSizeY, int filePart) {} - - public static MinDimEntry getMinimalEntry(ModuloDimensionEntries entry) { - long filePosition = entry.getFilePosition(); - int dimensionStartX = entry.getDimension("X").start; - int dimensionStartY = entry.getDimension("Y").start; - int dimensionStartZ = 0; - if (entry.hasDimension("Z")) { - dimensionStartZ = entry.getDimension("Z").start; - } - int storedSizeX = entry.getDimension("X").storedSize; - int storedSizeY = entry.getDimension("Y").storedSize; - int filePart = entry.filePart; - return new MinDimEntry(dimensionStartZ, filePosition, dimensionStartX, dimensionStartY, storedSizeX, storedSizeY, filePart); - }*/ - /** Duplicates this reader for parallel reading. * Creating a reader with this method allows to keep a very low memory footprint * because all immutable objects are re-used by reference. @@ -2100,7 +2211,7 @@ public ZeissCZIReader copy() { */ private static class CZISegments { final LibCZI.FileHeaderSegment fileHeader; - final LibCZI.SubBlockDirectorySegment subBlockDirectory; + final LibCZI.SubBlockDirectorySegment subBlockDirectory; // Some multipart files could have a part without this segment final LibCZI.AttachmentDirectorySegment attachmentDirectory; final LibCZI.MetaDataSegment metadata; final double[] timeStamps; @@ -2111,22 +2222,22 @@ public CZISegments(String id, boolean littleEndian) throws IOException { this.subBlockDirectory = LibCZI.getSubBlockDirectorySegment(this.fileHeader.data.subBlockDirectoryPosition, id, BUFFER_SIZE, littleEndian); this.attachmentDirectory = LibCZI.getAttachmentDirectorySegment(this.fileHeader.data.attachmentDirectoryPosition, id, BUFFER_SIZE, littleEndian); - // For searching of blocks at the end of the file, in case the metadata subblock has been deleted - long lastBlockPosition = LibCZI.getPositionLastBlock(subBlockDirectory); + // For searching of blocks at the end of the file, just in case the metadata subblock has been deleted + long lastBlockPosition = subBlockDirectory!=null?LibCZI.getPositionLastBlock(subBlockDirectory):0; this.metadata = LibCZI.getMetaDataSegment(this.fileHeader.data.metadataPosition, id, BUFFER_SIZE, littleEndian, lastBlockPosition); if (attachmentDirectory!=null) { this.timeStamps = LibCZI.getTimeStamps(this.attachmentDirectory, id, BUFFER_SIZE, littleEndian); - //System.out.println("#ts="+timeStamps.length); - /*for (double timeStamp: timeStamps) { - System.out.println(timeStamp); - }*/ } else { this.timeStamps = new double[0]; } } } + /** + * A least recently used cache for CZI subblocks. Its goal is to prevent multiple decompression of the same subblock + * when only subregions are requested. + */ static class SubBlockLRUCache extends LinkedHashMap> { @@ -2164,7 +2275,6 @@ protected boolean removeEldestEntry( totalWeight.addAndGet(-cost.get(eldest.getKey())); cost.remove(eldest.getKey()); eldest.getValue().clear(); - //System.out.println("Remove"); return true; } else return false; @@ -2175,10 +2285,9 @@ synchronized public void touch(final MinDimEntry key, { final SoftReference ref = get(key); if (ref == null) { - long costValue = value.length;//getWeight(value); + long costValue = value.length; totalWeight.addAndGet(costValue); cost.put(key, costValue); - //System.out.println(totalWeight.get()/(1024*1024)+" Mb"); put(key, new SoftReference<>(value)); } else if (ref.get() == null) { @@ -3237,8 +3346,10 @@ Corner fromBlocks(Collection blocks, int iCoreIndex, Unit u if ((x == null)||(x.value().doubleValue()>posX.value(unitLength).doubleValue())) { x = posX; } + + // UnitLength in Y may be Reference Frames for Line scans Length posY = new Length((double) iBlock.dimensionStartY/(double) reader.coreIndexToDownscaleFactor.get(iCoreIndex) - *coreToPixSizeY.get(iCoreIndex).value(unitLength).doubleValue(), unitLength); + *coreToPixSizeY.get(iCoreIndex).value(coreToPixSizeY.get(iCoreIndex).unit()).doubleValue(), coreToPixSizeY.get(iCoreIndex).unit()); if ((y == null)||(y.value().doubleValue()>posY.value(unitLength).doubleValue())) { y = posY; } @@ -3285,9 +3396,20 @@ private void setSpaceAndTimeInformation( // of series and of planes double cornerYAllScenesMicrons = Double.NaN; // Attempt strategy 1: - // - Let's pick the first subblock of the first core index + // - Let's pick the first subblock of the first core index, of the lowest czt key + // - We do not take CZTKey(0,0,0) because some files do not have this block (Zenodo repo 10577621 file Intestine_3color_RAC.czi) + int coreUsedForXYOffset = 0; - MinDimEntry firstSubBlock = mapCoreCZTToBlocks.get(coreUsedForXYOffset).get(new CZTKey(0, 0, 0)).get(0); + + final Comparator keyComparator = Comparator.comparingInt((CZTKey p) -> p.t) + .thenComparingInt(p -> p.c) + .thenComparingInt(p -> p.z); + + HashMap> blocksForXYOffset = mapCoreCZTToBlocks.get(coreUsedForXYOffset); + CZTKey cztKeyForXYOffset = blocksForXYOffset.keySet().stream().min(keyComparator::compare).get(); + + MinDimEntry firstSubBlock = blocksForXYOffset + .get(cztKeyForXYOffset).get(0); LibCZI.SubBlockSegment block = LibCZI.getBlock(reader.getStream(firstSubBlock.filePart), firstSubBlock.filePosition); LibCZI.SubBlockMeta sbm = LibCZI.readSubBlockMeta(reader.getStream(firstSubBlock.filePart), block, parser); @@ -3409,11 +3531,12 @@ private void setSpaceAndTimeInformation( // of series and of planes Length planePosX, planePosY, planePosZ = null; // plane position of the current coreindex - do not vary over z and t, but that could happen - blocks = mapCoreCZTToBlocks.get(iCoreIndex).get(new CZTKey(0,0,0)); + CZTKey cztKeySeriesForXYOffset = mapCoreCZTToBlocks.get(iCoreIndex).keySet().stream().min(keyComparator::compare).get(); + blocks = mapCoreCZTToBlocks.get(iCoreIndex).get(cztKeySeriesForXYOffset); if (!resolutionLevel0) { // Use the same position as the higher resolution level - // Keeping the last highest resolutio works because bio-formats forces the resolution + // Keeping the last highest resolution works because bio-formats forces the resolution // level series to be sorted according to the core series index: // res 0 series i / res 1 series i / res 2 series i / res 0 series i+1 / res 1 series i+1 etc. planePosX = planePosXResolutionLevel0; @@ -3665,8 +3788,20 @@ private void setSpaceAndTimeInformation( // of series and of planes } private void translateScaling(Element root) { + + // Default values in case no metadata is found : avoids NPE + for (int iCoreIndex=0; iCoreIndex 6) { color = color.substring(2); } @@ -4740,13 +4866,4 @@ private boolean isLatticeLightSheet() { } - static private String normalizeColor(String color) { - String c = color.replaceAll("#", ""); - if (c.length() > 6) { - c = c.substring(2, (int) Math.min(8, c.length())); - LOGGER.debug("Replaced color {} with {}", color, c); - } - return c; - } - } diff --git a/components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java b/components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java index 1203335cd26..eea309ee7ec 100644 --- a/components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java +++ b/components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java @@ -120,7 +120,8 @@ public static SubBlockDirectorySegment getSubBlockDirectorySegment(long blockPos } return directorySegment; } else { - throw new IOException(ZISRAWDIRECTORY+" segment expected, found "+segmentID+" instead."); + logger.warn(ZISRAWDIRECTORY+" segment expected, found "+segmentID+" instead."); + return null; // Some multipart file could be deprived of this segment. The reader needs to deal with null in this case } } } @@ -159,13 +160,12 @@ public static AttachmentDirectorySegment getAttachmentDirectorySegment(long bloc } else { logger.warn("No "+ZISRAWATTDIR+" segment found."); return null; - //throw new IOException(ZISRAWATTDIR+" segment expected, found "+segmentID+" instead."); } } } /** - * @param blockPosition position of the expected MetaData Segment + * @param blockPosition position of th file header segment of the referenced czi file * @param id a czi file path * @param BUFFER_SIZE the size of the caching buffer in bytes * @param isLittleEndian endianness of the data @@ -193,6 +193,7 @@ public static MetaDataSegment getMetaDataSegment(long blockPosition, String id, } else { metadataSegmentFound = true; } + if (metadataSegmentFound) { MetaDataSegment metaDataSegment = new MetaDataSegment(); // read the segment header @@ -224,6 +225,32 @@ private static AttachmentDirectorySegment.AttachmentDirectorySegmentData.Attachm return entry; } + // If blocks have been deleted, they are appended at the end of the file. This method looks for the location of the last known block and + // iterates over the next blocks until the end of the file. This method returns whether some orphan blocks have been found + static long findOrphanBlock(String id, int BUFFER_SIZE, boolean isLittleEndian, long positionOfLastKnownBlock, String blockType) throws IOException { + try (RandomAccessInputStream in = new RandomAccessInputStream(id, BUFFER_SIZE)) { + long eof = in.length(); + in.order(isLittleEndian); + long positionBlockStart = positionOfLastKnownBlock; + in.seek(positionBlockStart); + String segmentID = in.readString(16).trim(); + + long allocatedSize = in.readLong(); + long usedSize = in.readLong(); + while (positionBlockStart + allocatedSize + 32 Date: Thu, 22 Feb 2024 17:55:14 +0000 Subject: [PATCH 14/16] Update LibCZI.java with extra checks to avoid exceptions --- .../formats-gpl/src/loci/formats/in/libczi/LibCZI.java | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java b/components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java index eea309ee7ec..db659fb4715 100644 --- a/components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java +++ b/components/formats-gpl/src/loci/formats/in/libczi/LibCZI.java @@ -448,7 +448,9 @@ private static void skipEntryDV(RandomAccessInputStream in) throws IOException{ */ public static SubBlockSegment getBlock(RandomAccessInputStream in, long filePosition) throws IOException { SubBlockSegment subBlock = new SubBlockSegment(); - + if (filePosition + HEADER_SIZE + 16 + subBlock.data.metadataSize > in.length()) { + return null; + } in.seek(filePosition // Jumps 16 bytes to avoid reading the id, which should be ZISRAWSUBBLOCK anyway // Jumps 8 bytes for used size @@ -480,7 +482,8 @@ public static SubBlockSegment getBlock(RandomAccessInputStream in, long filePosi */ public static SubBlockMeta readSubBlockMeta(RandomAccessInputStream in, SubBlockSegment subBlock, DocumentBuilder parser) throws IOException { SubBlockMeta subBlockMeta = new SubBlockMeta(); - if (subBlock.dataOffset + subBlock.data.dataSize + subBlock.data.attachmentSize < in.length()) { + if (subBlock.dataOffset + subBlock.data.dataSize + subBlock.data.attachmentSize < in.length() && + subBlock.data.metadataOffset + subBlock.data.metadataSize < in.length() && subBlock.data.metadataSize > 0) { in.seek(subBlock.data.metadataOffset); //System.out.println("Offs= "+subBlock.data.metadataOffset); From b0cc110a18086a45c993de923f7077901c9031cc Mon Sep 17 00:00:00 2001 From: David Gault Date: Thu, 22 Feb 2024 17:55:50 +0000 Subject: [PATCH 15/16] Update ZeissCZIReader.java --- components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java | 1 + 1 file changed, 1 insertion(+) diff --git a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java index 7e6b43a9ad8..06027bfa88c 100644 --- a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java +++ b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java @@ -3300,6 +3300,7 @@ private LibCZI.SubBlockMeta getSubBlockMeta(int c, int z, int t, int coreIdx, List blocks = mapCoreCZTToBlocks.get(coreIdx).get(czt); if ((blocks==null) || (blocks.size()==0)) return null; LibCZI.SubBlockSegment block = LibCZI.getBlock(reader.getStream(blocks.get(0).filePart), blocks.get(0).filePosition); + if (block == null) return null; return LibCZI.readSubBlockMeta(reader.getStream(blocks.get(0).filePart), block, parser); } From 655062632b61d7e17238220340644d9a16a7bb52 Mon Sep 17 00:00:00 2001 From: David Gault Date: Fri, 23 Feb 2024 14:33:23 +0000 Subject: [PATCH 16/16] Update ZeissCZIReader.java Attempt to close open file handles --- .../formats-gpl/src/loci/formats/in/ZeissCZIReader.java | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java index 06027bfa88c..69616aac5d7 100644 --- a/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java +++ b/components/formats-gpl/src/loci/formats/in/ZeissCZIReader.java @@ -933,6 +933,9 @@ private synchronized RandomAccessInputStream getStream(int filePart) throws IOEx if ((in != null)&&(streamCurrentPart == filePart)) { return in; } + if (in != null) { + in.close(); + } streamCurrentPart = filePart; RandomAccessInputStream ris = new RandomAccessInputStream(filePartToFileName.get(filePart), BUFFER_SIZE); in = ris; @@ -1555,6 +1558,8 @@ private void addLabelIfExists(List sortedFileParts, Map 1 || c.sizeT > 1) { + stream.close(); + labelReader.close(); return; } core.add(new CoreMetadata(c)); @@ -1594,6 +1599,8 @@ private void addSlidePreviewIfExists(List sortedFileParts, Map 1 || c.sizeT > 1) { + stream.close(); + labelReader.close(); return; } core.add(new CoreMetadata(c));