From 983e073a64a4edea5f1343bd0648e626eaf78cd3 Mon Sep 17 00:00:00 2001 From: Louis Bergelson Date: Fri, 2 Jun 2023 15:50:16 -0400 Subject: [PATCH] Enable IntelInflater (#1885) * Fix longstanding bug so we use the Intel Inflater when it is requested * We have been using the Java Inflater even when the Intel one was available and requested due to a bug in the configuration of the default inflater. The fix is fairly brittle because it depends on the order of initialization of various things. * This will be fixed more generally by https://github.com/samtools/htsjdk/issues/1666 * Add a test --- .../picard/cmdline/CommandLineProgram.java | 5 +- .../picard/IntelInflaterDeflaterLoadTest.java | 57 +++++++++++++++++++ 2 files changed, 61 insertions(+), 1 deletion(-) diff --git a/src/main/java/picard/cmdline/CommandLineProgram.java b/src/main/java/picard/cmdline/CommandLineProgram.java index bd33cedce2..b86b82d700 100644 --- a/src/main/java/picard/cmdline/CommandLineProgram.java +++ b/src/main/java/picard/cmdline/CommandLineProgram.java @@ -218,7 +218,6 @@ public int instanceMain(final String[] argv) { this.defaultHeaders.add(new StringHeader("Started on: " + startDate)); Log.setGlobalLogLevel(VERBOSITY); - SamReaderFactory.setDefaultValidationStringency(VALIDATION_STRINGENCY); // Set the compression level everywhere we can think of BlockCompressedOutputStream.setDefaultCompressionLevel(COMPRESSION_LEVEL); @@ -257,6 +256,10 @@ public int instanceMain(final String[] argv) { BlockGunzipper.setDefaultInflaterFactory(new IntelInflaterFactory()); } + // This has to happen after the inflater factory is set because it causes a reinitialization of the static + // default reader factory. At least until https://github.com/samtools/htsjdk/issues/1666 is resolved + SamReaderFactory.setDefaultValidationStringency(VALIDATION_STRINGENCY); + if (!QUIET) { System.err.println("[" + new Date() + "] " + commandLine); diff --git a/src/test/java/picard/IntelInflaterDeflaterLoadTest.java b/src/test/java/picard/IntelInflaterDeflaterLoadTest.java index 8875c966c6..8db09ae8f2 100644 --- a/src/test/java/picard/IntelInflaterDeflaterLoadTest.java +++ b/src/test/java/picard/IntelInflaterDeflaterLoadTest.java @@ -1,11 +1,23 @@ package picard; import com.intel.gkl.compression.IntelDeflater; +import com.intel.gkl.compression.IntelDeflaterFactory; import com.intel.gkl.compression.IntelInflater; +import com.intel.gkl.compression.IntelInflaterFactory; +import htsjdk.samtools.SAMFileWriterFactory; +import htsjdk.samtools.SamReaderFactory; +import htsjdk.samtools.util.zip.DeflaterFactory; +import htsjdk.samtools.util.zip.InflaterFactory; import org.apache.commons.lang3.SystemUtils; +import org.broadinstitute.barclay.argparser.CommandLineProgramProperties; import org.testng.Assert; import org.testng.SkipException; import org.testng.annotations.Test; +import picard.cmdline.CommandLineProgram; +import picard.cmdline.programgroups.OtherProgramGroup; +import picard.cmdline.programgroups.Testing; + +import java.lang.reflect.Field; /** * Test that the Intel Inflater and Deflater can be loaded successfully. @@ -25,6 +37,20 @@ public void testIntelDeflaterIsAvailable() { "Intel shared library was not loaded. This could be due to a configuration error, or your system might not support it."); } + @Test + public void testIntelInflaterIsUsed(){ + final InflaterDeflaterTester cmd = new InflaterDeflaterTester(); + cmd.instanceMain(new String[]{}); + Assert.assertEquals(cmd.inflaterFactory.getClass(), IntelInflaterFactory.class); + } + + @Test + public void testDeflaterIsUsed(){ + final InflaterDeflaterTester cmd = new InflaterDeflaterTester(); + cmd.instanceMain(new String[]{}); + Assert.assertEquals(cmd.deflaterFactory.getClass(), IntelDeflaterFactory.class); + } + private void checkIntelSupported(final String componentName) { if (!SystemUtils.IS_OS_LINUX && !SystemUtils.IS_OS_MAC) { throw new SkipException(componentName + " is not available on this platform"); @@ -34,4 +60,35 @@ private void checkIntelSupported(final String componentName) { throw new SkipException(componentName + " is not available for this architecture"); } } + + + @CommandLineProgramProperties(summary = "test program for checking if the intel optimized inflater/deflater are active", + oneLineSummary = "test program please ignore", + programGroup = Testing.class, + omitFromCommandLine = true) + public static class InflaterDeflaterTester extends CommandLineProgram { + public InflaterFactory inflaterFactory; + public DeflaterFactory deflaterFactory; + + @Override + protected int doWork() { + final SamReaderFactory samReaderFactory = SamReaderFactory.makeDefault(); + inflaterFactory = getFieldValue(samReaderFactory, "inflaterFactory", InflaterFactory.class); + + final SAMFileWriterFactory samFileWriterFactory = new SAMFileWriterFactory(); + deflaterFactory = getFieldValue(samFileWriterFactory, "deflaterFactory", DeflaterFactory.class); + + return 0; + } + + private R getFieldValue(final T obj,final String fieldName, Class clazz) { + try { + final Field deflaterFactoryField = obj.getClass().getDeclaredField(fieldName); + deflaterFactoryField.setAccessible(true); + return clazz.cast(deflaterFactoryField.get(obj)); + } catch (NoSuchFieldException | IllegalAccessException e) { + throw new RuntimeException(e); + } + } + } }