diff --git a/.github/native-tests.json b/.github/native-tests.json
index f738e83fdfe7b..1a803bf484efc 100644
--- a/.github/native-tests.json
+++ b/.github/native-tests.json
@@ -143,6 +143,12 @@
"timeout": 70,
"test-modules": "devtools-registry-client",
"os-name": "ubuntu-latest"
- }
+ },
+ {
+ "category": "AWT, ImageIO and Java2D",
+ "timeout": 20,
+ "test-modules": "awt",
+ "os-name": "ubuntu-latest"
+ }
]
}
diff --git a/bom/application/pom.xml b/bom/application/pom.xml
index 2d7d06e0cacbc..1b85634ade093 100644
--- a/bom/application/pom.xml
+++ b/bom/application/pom.xml
@@ -5414,6 +5414,16 @@
quarkus-vertx-web-deployment
${project.version}
+
+ io.quarkus
+ quarkus-awt
+ ${project.version}
+
+
+ io.quarkus
+ quarkus-awt-deployment
+ ${project.version}
+
diff --git a/core/runtime/src/main/java/io/quarkus/runtime/graal/AwtFeature.java b/core/runtime/src/main/java/io/quarkus/runtime/graal/AwtFeature.java
new file mode 100644
index 0000000000000..2e71316fdb9b7
--- /dev/null
+++ b/core/runtime/src/main/java/io/quarkus/runtime/graal/AwtFeature.java
@@ -0,0 +1,34 @@
+package io.quarkus.runtime.graal;
+
+import org.graalvm.nativeimage.ImageSingletons;
+import org.graalvm.nativeimage.hosted.Feature;
+import org.graalvm.nativeimage.impl.RuntimeClassInitializationSupport;
+
+import com.oracle.svm.core.annotate.AutomaticFeature;
+
+/**
+ * Technically, this should live in extensions/awt,
+ * but currently all code that relies on JAXB
+ * requires at the very least sun.java2d package to be runtime initialized.
+ *
+ * Having sun.java2d code initialized at build time caused issues,
+ * which is why a substitution was set in place to avoid such code making it to the binary:
+ * https://github.com/quarkusio/quarkus/commit/ef87e5567cf3ac462a3f12aad4b5b530d9220223
+ *
+ * So, as long as JAXB graphics code has not been excluded completely from JAXB,
+ * it is safer to define all image related packages to be runtime initialized directly in core.
+ */
+@AutomaticFeature
+public class AwtFeature implements Feature {
+ @Override
+ public void afterRegistration(AfterRegistrationAccess access) {
+ final RuntimeClassInitializationSupport runtimeInit = ImageSingletons.lookup(RuntimeClassInitializationSupport.class);
+ final String reason = "Quarkus run time init for AWT";
+ runtimeInit.initializeAtRunTime("com.sun.imageio", reason);
+ runtimeInit.initializeAtRunTime("java.awt", reason);
+ runtimeInit.initializeAtRunTime("javax.imageio", reason);
+ runtimeInit.initializeAtRunTime("sun.awt", reason);
+ runtimeInit.initializeAtRunTime("sun.font", reason);
+ runtimeInit.initializeAtRunTime("sun.java2d", reason);
+ }
+}
diff --git a/core/runtime/src/main/java/io/quarkus/runtime/graal/Java2DSubstitutions.java b/core/runtime/src/main/java/io/quarkus/runtime/graal/Java2DSubstitutions.java
deleted file mode 100644
index 0ca2b30ab278e..0000000000000
--- a/core/runtime/src/main/java/io/quarkus/runtime/graal/Java2DSubstitutions.java
+++ /dev/null
@@ -1,20 +0,0 @@
-package io.quarkus.runtime.graal;
-
-import java.awt.Graphics;
-import java.awt.GraphicsEnvironment;
-
-import com.oracle.svm.core.annotate.AlwaysInline;
-import com.oracle.svm.core.annotate.Substitute;
-import com.oracle.svm.core.annotate.TargetClass;
-
-@TargetClass(GraphicsEnvironment.class)
-final class Target_java_awt_GraphicsEnvironment {
- @AlwaysInline("DCE for things using Java2D")
- @Substitute
- public static Graphics getLocalGraphicsEnvironment() {
- throw new UnsupportedOperationException("Not implemented yet for GraalVM native images");
- }
-}
-
-class Java2DSubstitutions {
-}
diff --git a/devtools/bom-descriptor-json/pom.xml b/devtools/bom-descriptor-json/pom.xml
index 75612efc9853c..f919a05eaaed7 100644
--- a/devtools/bom-descriptor-json/pom.xml
+++ b/devtools/bom-descriptor-json/pom.xml
@@ -383,6 +383,19 @@
+
+ io.quarkus
+ quarkus-awt
+ ${project.version}
+ pom
+ test
+
+
+ *
+ *
+
+
+
io.quarkus
quarkus-azure-functions-http
diff --git a/docs/pom.xml b/docs/pom.xml
index 610684e926b42..e505d8a052b3f 100644
--- a/docs/pom.xml
+++ b/docs/pom.xml
@@ -344,6 +344,19 @@
+
+ io.quarkus
+ quarkus-awt-deployment
+ ${project.version}
+ pom
+ test
+
+
+ *
+ *
+
+
+
io.quarkus
quarkus-azure-functions-http-deployment
diff --git a/extensions/awt/deployment/pom.xml b/extensions/awt/deployment/pom.xml
new file mode 100644
index 0000000000000..a31abf40b9e4a
--- /dev/null
+++ b/extensions/awt/deployment/pom.xml
@@ -0,0 +1,43 @@
+
+
+ 4.0.0
+
+ io.quarkus
+ quarkus-awt-parent
+ 999-SNAPSHOT
+
+ quarkus-awt-deployment
+ Quarkus - Awt - Deployment
+
+
+ io.quarkus
+ quarkus-arc-deployment
+
+
+ io.quarkus
+ quarkus-awt
+
+
+ io.quarkus
+ quarkus-junit5-internal
+ test
+
+
+
+
+
+ maven-compiler-plugin
+
+
+
+ io.quarkus
+ quarkus-extension-processor
+ ${project.version}
+
+
+
+
+
+
+
diff --git a/extensions/awt/deployment/src/main/java/io/quarkus/awt/deployment/AwtProcessor.java b/extensions/awt/deployment/src/main/java/io/quarkus/awt/deployment/AwtProcessor.java
new file mode 100644
index 0000000000000..f41aadc2a7916
--- /dev/null
+++ b/extensions/awt/deployment/src/main/java/io/quarkus/awt/deployment/AwtProcessor.java
@@ -0,0 +1,75 @@
+package io.quarkus.awt.deployment;
+
+import io.quarkus.deployment.annotations.BuildStep;
+import io.quarkus.deployment.builditem.FeatureBuildItem;
+import io.quarkus.deployment.builditem.nativeimage.JniRuntimeAccessBuildItem;
+import io.quarkus.deployment.builditem.nativeimage.ReflectiveClassBuildItem;
+
+class AwtProcessor {
+ private static final String FEATURE = "awt";
+
+ @BuildStep
+ ReflectiveClassBuildItem setupReflectionClasses() {
+ return new ReflectiveClassBuildItem(false, false,
+ "sun.awt.X11GraphicsEnvironment",
+ "sun.awt.X11.XToolkit");
+ }
+
+ @BuildStep
+ JniRuntimeAccessBuildItem setupJava2DClasses() {
+ return new JniRuntimeAccessBuildItem(true, true, true,
+ "java.awt.AlphaComposite",
+ "java.awt.Color",
+ "java.awt.geom.AffineTransform",
+ "java.awt.geom.Path2D",
+ "java.awt.geom.Path2D$Float",
+ "java.awt.image.BufferedImage",
+ "java.awt.image.ColorModel",
+ "java.awt.image.IndexColorModel",
+ "java.awt.image.Raster",
+ "java.awt.image.SampleModel",
+ "java.awt.image.SinglePixelPackedSampleModel",
+ "sun.awt.SunHints",
+ "sun.awt.image.BufImgSurfaceData$ICMColorData",
+ "sun.awt.image.ByteComponentRaster",
+ "sun.awt.image.BytePackedRaster",
+ "sun.awt.image.ImageRepresentation",
+ "sun.awt.image.IntegerComponentRaster",
+ "sun.java2d.Disposer",
+ "sun.java2d.InvalidPipeException",
+ "sun.java2d.NullSurfaceData",
+ "sun.java2d.SunGraphics2D",
+ "sun.java2d.SurfaceData",
+ "sun.java2d.loops.Blit",
+ "sun.java2d.loops.BlitBg",
+ "sun.java2d.loops.CompositeType",
+ "sun.java2d.loops.DrawGlyphList",
+ "sun.java2d.loops.DrawGlyphListAA",
+ "sun.java2d.loops.DrawGlyphListLCD",
+ "sun.java2d.loops.DrawLine",
+ "sun.java2d.loops.DrawParallelogram",
+ "sun.java2d.loops.DrawPath",
+ "sun.java2d.loops.DrawPolygons",
+ "sun.java2d.loops.DrawRect",
+ "sun.java2d.loops.FillParallelogram",
+ "sun.java2d.loops.FillPath",
+ "sun.java2d.loops.FillRect",
+ "sun.java2d.loops.FillSpans",
+ "sun.java2d.loops.GraphicsPrimitive",
+ "sun.java2d.loops.GraphicsPrimitiveMgr",
+ "sun.java2d.loops.GraphicsPrimitive[]",
+ "sun.java2d.loops.MaskBlit",
+ "sun.java2d.loops.MaskFill",
+ "sun.java2d.loops.ScaledBlit",
+ "sun.java2d.loops.SurfaceType",
+ "sun.java2d.loops.TransformHelper",
+ "sun.java2d.loops.XORComposite",
+ "sun.java2d.pipe.Region",
+ "sun.java2d.pipe.RegionIterator");
+ }
+
+ @BuildStep
+ FeatureBuildItem feature() {
+ return new FeatureBuildItem(FEATURE);
+ }
+}
diff --git a/extensions/awt/pom.xml b/extensions/awt/pom.xml
new file mode 100644
index 0000000000000..e28ae5059bd3f
--- /dev/null
+++ b/extensions/awt/pom.xml
@@ -0,0 +1,17 @@
+
+
+ 4.0.0
+
+ io.quarkus
+ quarkus-extensions-parent
+ 999-SNAPSHOT
+
+ quarkus-awt-parent
+ pom
+ Quarkus - Awt - Parent
+
+ deployment
+ runtime
+
+
diff --git a/extensions/awt/runtime/pom.xml b/extensions/awt/runtime/pom.xml
new file mode 100644
index 0000000000000..e4ce9c74395b2
--- /dev/null
+++ b/extensions/awt/runtime/pom.xml
@@ -0,0 +1,44 @@
+
+
+ 4.0.0
+
+ io.quarkus
+ quarkus-awt-parent
+ 999-SNAPSHOT
+
+ quarkus-awt
+ Quarkus - Awt - Runtime
+ Enable AWT and Java2D usage
+
+
+ io.quarkus
+ quarkus-arc
+
+
+ org.graalvm.nativeimage
+ svm
+ provided
+
+
+
+
+
+ io.quarkus
+ quarkus-bootstrap-maven-plugin
+
+
+ maven-compiler-plugin
+
+
+
+ io.quarkus
+ quarkus-extension-processor
+ ${project.version}
+
+
+
+
+
+
+
diff --git a/extensions/awt/runtime/src/main/resources/META-INF/quarkus-extension.yaml b/extensions/awt/runtime/src/main/resources/META-INF/quarkus-extension.yaml
new file mode 100644
index 0000000000000..cebd019240806
--- /dev/null
+++ b/extensions/awt/runtime/src/main/resources/META-INF/quarkus-extension.yaml
@@ -0,0 +1,12 @@
+name: AWT
+metadata:
+ keywords:
+ - awt
+ - font
+ - java2d
+ - image
+ - imageio
+# guide: ...
+ categories:
+ - "miscellaneous"
+ status: "preview"
\ No newline at end of file
diff --git a/extensions/pom.xml b/extensions/pom.xml
index 86697504fe80d..3ee41c6ef7cbd 100644
--- a/extensions/pom.xml
+++ b/extensions/pom.xml
@@ -201,6 +201,8 @@
vertx-web
+
+ awt
diff --git a/extensions/tika/deployment/pom.xml b/extensions/tika/deployment/pom.xml
index d2a1bba4af648..640605ff71659 100644
--- a/extensions/tika/deployment/pom.xml
+++ b/extensions/tika/deployment/pom.xml
@@ -20,6 +20,10 @@
io.quarkus
quarkus-arc-deployment
+
+ io.quarkus
+ quarkus-awt-deployment
+
io.quarkus
diff --git a/extensions/tika/runtime/pom.xml b/extensions/tika/runtime/pom.xml
index 9f48205241c75..99e00f57b6f94 100644
--- a/extensions/tika/runtime/pom.xml
+++ b/extensions/tika/runtime/pom.xml
@@ -20,6 +20,10 @@
io.quarkus
quarkus-arc
+
+ io.quarkus
+ quarkus-awt
+
io.quarkus
diff --git a/extensions/tika/runtime/src/main/java/io/quarkus/tika/runtime/graal/TikaFeature.java b/extensions/tika/runtime/src/main/java/io/quarkus/tika/runtime/graal/TikaFeature.java
new file mode 100644
index 0000000000000..cea8d193bbc3a
--- /dev/null
+++ b/extensions/tika/runtime/src/main/java/io/quarkus/tika/runtime/graal/TikaFeature.java
@@ -0,0 +1,19 @@
+package io.quarkus.tika.runtime.graal;
+
+import org.graalvm.nativeimage.ImageSingletons;
+import org.graalvm.nativeimage.hosted.Feature;
+import org.graalvm.nativeimage.impl.RuntimeClassInitializationSupport;
+
+import com.oracle.svm.core.annotate.AutomaticFeature;
+
+@AutomaticFeature
+public class TikaFeature implements Feature {
+ @Override
+ public void afterRegistration(AfterRegistrationAccess access) {
+ final RuntimeClassInitializationSupport runtimeInit = ImageSingletons.lookup(RuntimeClassInitializationSupport.class);
+ final String reason = "Quarkus run time init for Apache Tika";
+ runtimeInit.initializeAtRunTime("org.apache.pdfbox", reason);
+ runtimeInit.initializeAtRunTime("org.apache.poi.hssf.util", reason);
+ runtimeInit.initializeAtRunTime("org.apache.poi.ss.format", reason);
+ }
+}
diff --git a/integration-tests/awt/pom.xml b/integration-tests/awt/pom.xml
new file mode 100644
index 0000000000000..2f29b6b7716e4
--- /dev/null
+++ b/integration-tests/awt/pom.xml
@@ -0,0 +1,93 @@
+
+
+ 4.0.0
+
+ io.quarkus
+ quarkus-integration-tests-parent
+ 999-SNAPSHOT
+
+ quarkus-awt-integration-tests
+ Quarkus - Awt - Integration Tests
+
+
+
+ io.quarkus
+ quarkus-awt-deployment
+ ${project.version}
+ pom
+ test
+
+
+ *
+ *
+
+
+
+
+ io.quarkus
+ quarkus-awt
+
+
+ io.quarkus
+ quarkus-junit5
+ test
+
+
+
+
+
+ io.quarkus
+ quarkus-maven-plugin
+
+
+
+ build
+
+
+
+
+
+
+
+
+ native-image
+
+
+ native
+
+
+
+
+
+ maven-surefire-plugin
+
+ ${native.surefire.skip}
+
+
+
+ maven-failsafe-plugin
+
+
+
+ integration-test
+ verify
+
+
+
+ ${project.build.directory}/${project.build.finalName}-runner
+ org.jboss.logmanager.LogManager
+ ${maven.home}
+
+
+
+
+
+
+
+
+ native
+
+
+
+
diff --git a/integration-tests/awt/src/main/java/io/quarkus/awt/it/Dummy.java b/integration-tests/awt/src/main/java/io/quarkus/awt/it/Dummy.java
new file mode 100644
index 0000000000000..894f4a92f03e9
--- /dev/null
+++ b/integration-tests/awt/src/main/java/io/quarkus/awt/it/Dummy.java
@@ -0,0 +1,5 @@
+package io.quarkus.awt.it;
+
+public class Dummy {
+ // Dummy class to have a jar file
+}
diff --git a/integration-tests/awt/src/main/resources/META-INF/resources/1px.png b/integration-tests/awt/src/main/resources/META-INF/resources/1px.png
new file mode 100644
index 0000000000000..c5916f2897056
Binary files /dev/null and b/integration-tests/awt/src/main/resources/META-INF/resources/1px.png differ
diff --git a/integration-tests/awt/src/test/java/io/quarkus/awt/it/AwtImageResizeIT.java b/integration-tests/awt/src/test/java/io/quarkus/awt/it/AwtImageResizeIT.java
new file mode 100644
index 0000000000000..9c39a3d380993
--- /dev/null
+++ b/integration-tests/awt/src/test/java/io/quarkus/awt/it/AwtImageResizeIT.java
@@ -0,0 +1,7 @@
+package io.quarkus.awt.it;
+
+import io.quarkus.test.junit.main.QuarkusMainIntegrationTest;
+
+@QuarkusMainIntegrationTest
+public class AwtImageResizeIT extends AwtImageResizeTest {
+}
diff --git a/integration-tests/awt/src/test/java/io/quarkus/awt/it/AwtImageResizeTest.java b/integration-tests/awt/src/test/java/io/quarkus/awt/it/AwtImageResizeTest.java
new file mode 100644
index 0000000000000..70f2f500e4888
--- /dev/null
+++ b/integration-tests/awt/src/test/java/io/quarkus/awt/it/AwtImageResizeTest.java
@@ -0,0 +1,51 @@
+package io.quarkus.awt.it;
+
+import java.awt.Image;
+import java.awt.image.BufferedImage;
+import java.io.ByteArrayInputStream;
+import java.io.ByteArrayOutputStream;
+import java.io.IOException;
+import java.util.Base64;
+
+import javax.imageio.ImageIO;
+
+import org.junit.jupiter.api.Assertions;
+import org.junit.jupiter.api.Test;
+
+import io.quarkus.test.junit.main.QuarkusMainTest;
+
+@QuarkusMainTest
+public class AwtImageResizeTest {
+
+ @Test
+ public void testImageResize() throws IOException {
+ final String imageData = "";
+ final String expected = "";
+
+ final byte[] imageBytes = Base64.getDecoder().decode(imageData.split(",", 2)[1]);
+ final String resizeImage = resizeImage(imageBytes, 100);
+ Assertions.assertEquals(expected, resizeImage);
+ }
+
+ private String resizeImage(byte[] imageBytes, int height) throws IOException {
+ BufferedImage inputImage = ImageIO.read(new ByteArrayInputStream(imageBytes));
+
+ int currentW = inputImage.getWidth();
+ int currentH = inputImage.getHeight();
+ int width = currentW * height / currentH;
+ if (currentH < height) {
+ width = currentW;
+ height = currentH;
+ }
+
+ Image originalImage = inputImage.getScaledInstance(width, height, Image.SCALE_SMOOTH);
+ BufferedImage resizedImage = new BufferedImage(width, height, BufferedImage.TYPE_3BYTE_BGR);
+ resizedImage.getGraphics().drawImage(originalImage, 0, 0, null);
+
+ ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
+ ImageIO.write(resizedImage, "png", outputStream);
+
+ return String.format("data:image/png;base64,%s", Base64.getEncoder().encodeToString(outputStream.toByteArray()));
+ }
+
+}
diff --git a/integration-tests/awt/src/test/java/io/quarkus/awt/it/ImageIOITCase.java b/integration-tests/awt/src/test/java/io/quarkus/awt/it/ImageIOITCase.java
new file mode 100644
index 0000000000000..3b2f0db61b7cc
--- /dev/null
+++ b/integration-tests/awt/src/test/java/io/quarkus/awt/it/ImageIOITCase.java
@@ -0,0 +1,7 @@
+package io.quarkus.awt.it;
+
+import io.quarkus.test.junit.main.QuarkusMainIntegrationTest;
+
+@QuarkusMainIntegrationTest
+public class ImageIOITCase extends ImageIOTestCase {
+}
diff --git a/integration-tests/awt/src/test/java/io/quarkus/awt/it/ImageIOTestCase.java b/integration-tests/awt/src/test/java/io/quarkus/awt/it/ImageIOTestCase.java
new file mode 100644
index 0000000000000..10736a232646d
--- /dev/null
+++ b/integration-tests/awt/src/test/java/io/quarkus/awt/it/ImageIOTestCase.java
@@ -0,0 +1,24 @@
+package io.quarkus.awt.it;
+
+import java.awt.image.BufferedImage;
+import java.io.IOException;
+
+import javax.imageio.ImageIO;
+
+import org.junit.jupiter.api.Assertions;
+import org.junit.jupiter.api.Test;
+
+import io.quarkus.test.junit.main.QuarkusMainTest;
+
+@QuarkusMainTest
+public class ImageIOTestCase {
+
+ @Test
+ public void testImageRead() throws IOException {
+ ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
+ BufferedImage image = ImageIO.read(classLoader.getResource("META-INF/resources/1px.png"));
+ Assertions.assertEquals(1, image.getHeight());
+ Assertions.assertEquals(1, image.getWidth());
+ }
+
+}
diff --git a/integration-tests/main/src/main/java/io/quarkus/it/corestuff/ImageIOSupport.java b/integration-tests/main/src/main/java/io/quarkus/it/corestuff/ImageIOSupport.java
deleted file mode 100644
index 18358d57acb01..0000000000000
--- a/integration-tests/main/src/main/java/io/quarkus/it/corestuff/ImageIOSupport.java
+++ /dev/null
@@ -1,22 +0,0 @@
-package io.quarkus.it.corestuff;
-
-import java.awt.image.BufferedImage;
-import java.io.IOException;
-
-import javax.imageio.ImageIO;
-import javax.servlet.annotation.WebServlet;
-import javax.servlet.http.HttpServlet;
-import javax.servlet.http.HttpServletRequest;
-import javax.servlet.http.HttpServletResponse;
-
-@WebServlet(name = "ImageIOTestEndpoint", urlPatterns = "/core/imageio")
-public class ImageIOSupport extends HttpServlet {
-
- @Override
- protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws IOException {
- ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
- BufferedImage image = ImageIO.read(classLoader.getResource("META-INF/resources/1px.png"));
- resp.getWriter().write(image.getHeight() + "x" + image.getWidth());
- }
-
-}
diff --git a/integration-tests/main/src/test/java/io/quarkus/it/main/ImageIOITCase.java b/integration-tests/main/src/test/java/io/quarkus/it/main/ImageIOITCase.java
deleted file mode 100644
index a97a1ea3974a0..0000000000000
--- a/integration-tests/main/src/test/java/io/quarkus/it/main/ImageIOITCase.java
+++ /dev/null
@@ -1,7 +0,0 @@
-package io.quarkus.it.main;
-
-import io.quarkus.test.junit.NativeImageTest;
-
-@NativeImageTest
-public class ImageIOITCase extends ImageIOTestCase {
-}
diff --git a/integration-tests/main/src/test/java/io/quarkus/it/main/ImageIOTestCase.java b/integration-tests/main/src/test/java/io/quarkus/it/main/ImageIOTestCase.java
deleted file mode 100644
index 66dbbc1eb55ee..0000000000000
--- a/integration-tests/main/src/test/java/io/quarkus/it/main/ImageIOTestCase.java
+++ /dev/null
@@ -1,18 +0,0 @@
-package io.quarkus.it.main;
-
-import static io.restassured.RestAssured.when;
-import static org.hamcrest.Matchers.equalTo;
-
-import org.junit.jupiter.api.Test;
-
-import io.quarkus.test.junit.QuarkusTest;
-
-@QuarkusTest
-public class ImageIOTestCase {
-
- @Test
- public void testImageRead() {
- when().get("/core/imageio").then().body(equalTo("1x1"));
- }
-
-}
diff --git a/integration-tests/pom.xml b/integration-tests/pom.xml
index b8a59a9881b89..e7649c044d4e2 100644
--- a/integration-tests/pom.xml
+++ b/integration-tests/pom.xml
@@ -114,6 +114,7 @@
avro-reload
+ awt
bouncycastle
bouncycastle-fips
bouncycastle-fips-jsse