[rfc][icedtea-web] manager for desktop integration
Jacob Wisor
gitne at gmx.de
Mon Nov 2 08:12:22 UTC 2015
On 10/29/2015 at 05:06 PM Jiri Vanek wrote:
>> [*]
>> in light of [**]..... I will go with the patch without ico support.
>> And will again[***] try spi provider. As it may be really usefull. And post
>> ico support as separate
>> patch. Lets see how it will be sucessful as.. bad luck... I already[***] tried
>> to add wraper about
>> this Ico.java providing spi support. After several hours I discarded it as not
>> necessary. But
>> because of [**] lets try again....
>>
>
> So I created spi for Ico (still based on the same file - one update - the
> example is no loner available on that page :D ( I guess they removed it when I
> asked them for usage)
See, so the issue of licensing cannot be simply tossed aside.
> Further.... Yes, ImageIO.read and *relatives* can read ico files. BUT eg
> suntoolkit.createImage can NOT. Thats a bit of betrayal because it and its
> *relatives* are used for reading and especially asynchronous reading of images
> pretty often. It can be traced in ITW itself on some places.
> Yes we can fix it in ITW, But eg jtextpane used in this original patch or other
> users of toolkit.createImage can not.
I am not sure I fully understand what you mean. Why is asynchronous reading of
images important per se? If you read images, or read a lot of any sort of data,
or do any other time consuming operations on the UI thread then you are doing
something wrong by design, regardless of the fact if you have an asynchronous
API at hand or not. Asynchronous APIs may help or ease programming on the UI
thread, and are even recommended to use if available, but they are not a
required for proper UI thread programming. You can (or should) always spawn
worker threads for blocking or time consuming operations on the UI thread. Of
course, asynchronous APIs implemented via interrupts or signals are always to be
preferred for the UI thread over asynchronous APIs implemented via worker threads.
But, I am not sure what you mean exactly, so I may have missed your point.
> if you will be wondering why it can not read it, you will end at:
>
>
> protected ImageDecoder getDecoder(InputStream is) {
> if (!is.markSupported())
> is = new BufferedInputStream(is);
> try {
> /* changed to support png
> is.mark(6);
> */
> is.mark(8);
> int c1 = is.read();
> int c2 = is.read();
> int c3 = is.read();
> int c4 = is.read();
> int c5 = is.read();
> int c6 = is.read();
> // added to support png
> int c7 = is.read();
> int c8 = is.read();
> // end of adding
> is.reset();
> is.mark(-1);
>
> if (c1 == 'G' && c2 == 'I' && c3 == 'F' && c4 == '8') {
> return new GifImageDecoder(this, is);
> } else if (c1 == '\377' && c2 == '\330' && c3 == '\377') {
> return new JPEGImageDecoder(this, is);
> } else if (c1 == '#' && c2 == 'd' && c3 == 'e' && c4 == 'f') {
> return new XbmImageDecoder(this, is);
> // } else if (c1 == '!' && c2 == ' ' && c3 == 'X' && c4 == 'P' &&
> // c5 == 'M' && c6 == '2') {
> // return new Xpm2ImageDecoder(this, is);
> // added to support png
> } else if (c1 == 137 && c2 == 80 && c3 == 78 &&
> c4 == 71 && c5 == 13 && c6 == 10 &&
> c7 == 26 && c8 == 10) {
> return new PNGImageDecoder(this, is);
> }
> // end of adding
> } catch (IOException e) {
> }
>
> return null;
> }
>
>
>
>
> Yes. reimplemented wheel in JDK itself :-///
Is this OpenJDK code? Because I am not sure what you are trying to say and are
presenting here.
Anyhow, if this is your code, and since you've been complaining that I never
make code suggestions, you may want to consider this for greatly improved
performance and efficiency:
public static final int GIF_MAGIC = 'G' << 24 | 'I' << 16 | 'F' << 8 | '8',
JPEG_MAGIC = 0xFFD8FF00,
XBM_MAGIC = '#' << 24 | 'd' << 16 | 'e' << 8 | 'f';
public static final long XPM2_MAGIC = '!' << 56 | ' ' << 48 | 'X' << 40 |
'P' << 32 | 'M' << 24 | '2' << 16,
PNG_MAGIC = (long)'\u0089' << 56 |
(long)'P' << 48 |
(long)'N' << 40 |
(long)'G' << 32 |
'\r' << 24 |
'\n' << 16 |
'\u001a' << 8 |
'\n';
protected ImageDecoder getDecoder(InputStream is) {
try {
if (!is.markSupported())
is = new BufferedInputStream(is);
/* changed to support png
is.mark(6);
*/
final DataInputStream dis = new DataInputStream(is);
dis.mark(8);
final long magic8 = dis.readLong();
dis.reset();
final int magic4;
if (((magic4 = (int)(magic8 >>> 32)) ^ GIF_MAGIC) == 0) {
// if (c1 == 'G' && c2 == 'I' && c3 == 'F' && c4 == '8') {
return new GifImageDecoder(this, is);
}
if ((magic4 & ~0xFF ^ JPEG_MAGIC) == 0) {
// } else if (c1 == '\377' && c2 == '\330' && c3 == '\377') {
return new JPEGImageDecoder(this, is);
}
if ((magic4 ^ XBM_MAGIC) == 0) {
// } else if (c1 == '#' && c2 == 'd' && c3 == 'e' && c4 == 'f') {
return new XbmImageDecoder(this, is);
}
if ((magic8 & ~0xFFFFL ^ XPM2_MAGIC) == 0L) {
// } else if (c1 == '!' && c2 == ' ' && c3 == 'X' && c4 == 'P' &&
// c5 == 'M' && c6 == '2') {
return new Xpm2ImageDecoder(this, is);
}
// added to support png
if ((magic8 ^ PNG_MAGIC) == 0L) {
// } else if (c1 == 137 && c2 == 80 && c3 == 78 &&
// c4 == 71 && c5 == 13 && c6 == 10 &&
// c7 == 26 && c8 == 10) {
return new PNGImageDecoder(this, is);
}
// end of adding
} catch (IOException e) {}
return null;
}
Besides, I think you may want to take a look at
javax.imageio.stream.FileImageInputStream. It's much more convenient and removes
the need for DataInputStream.
Generally speaking, testing ints or longs uses way less CPU cycles, less energy,
and is much faster than having to read single bytes, compare single bytes, and
finally test the AND logic (even tough it stops on first false). Most RISC
machines incur an additional load penalty for non-register word with operations,
unlike x86/x86_64. Especially ARM machines will suffer. And, although the
comparison is done at register width effectively (bytes, chars, shorts get
promoted or extended to register width) on RISC machines, the operands still
have to be loaded from potentially non-word aligned memory and then promoted to
register width. Even x86/x86_64 will suffer because of too many comparisons and
AND operations. Besides, all these comparisons are non-zero comparisons which
effectively disrupt jump prediction units in modern CPUs. This may seem like
being overly pedantic but these effects are clearly observable in Java too,
regardless if executing JIT compiled or interpreted code.
I looked at http://www.daubnet.com/en/file-format-ico and the Wikipedia, and as
far as I understand, ico can function as a container for other payloads. So, one
can have PNGs or JPEGs in an ico file, and as of Windows Vista this seams to be
part of the specification. Hence, any ico reader must be able to decode those
formats too. However, this list of acceptable payloads obviously also includes
BMP and the ico's original XOR encoding of BMP. So, in order to by fully
compliant (or legacy compatible in that sense) the decoder has to read BMP too.
AFAIK, OpenJDK does not support BMP. This may be a dead end unless you want to
implement a BMP decoder too.
> So what now?
I thought you said you would still try to implement an ico image decoder via
javax.imageio.spi?
Regards,
Jacob
More information about the distro-pkg-dev
mailing list