Pillow/src/PIL/Jpeg2KImagePlugin.py

315 lines
8.5 KiB
Python
Raw Normal View History

#
# The Python Imaging Library
# $Id$
#
# JPEG2000 file handling
#
# History:
# 2014-03-12 ajh Created
#
# Copyright (c) 2014 Coriolis Systems Limited
# Copyright (c) 2014 Alastair Houghton
#
# See the README file for information on usage and redistribution.
#
import io
import os
import struct
from . import Image, ImageFile
def _parse_codestream(fp):
"""Parse the JPEG 2000 codestream to extract the size and component
count from the SIZ marker segment, returning a PIL (size, mode) tuple."""
hdr = fp.read(2)
2019-03-21 16:28:20 +03:00
lsiz = struct.unpack(">H", hdr)[0]
siz = hdr + fp.read(lsiz - 2)
2019-03-21 16:28:20 +03:00
lsiz, rsiz, xsiz, ysiz, xosiz, yosiz, _, _, _, _, csiz = struct.unpack_from(
">HHIIIIIIIIH", siz
)
ssiz = [None] * csiz
xrsiz = [None] * csiz
yrsiz = [None] * csiz
for i in range(csiz):
2019-03-21 16:28:20 +03:00
ssiz[i], xrsiz[i], yrsiz[i] = struct.unpack_from(">BBB", siz, 36 + 3 * i)
size = (xsiz - xosiz, ysiz - yosiz)
if csiz == 1:
2019-03-21 16:28:20 +03:00
if (yrsiz[0] & 0x7F) > 8:
mode = "I;16"
else:
2019-03-21 16:28:20 +03:00
mode = "L"
elif csiz == 2:
2019-03-21 16:28:20 +03:00
mode = "LA"
elif csiz == 3:
2019-03-21 16:28:20 +03:00
mode = "RGB"
elif csiz == 4:
2019-03-21 16:28:20 +03:00
mode = "RGBA"
else:
mode = None
return (size, mode)
def _parse_jp2_header(fp):
"""Parse the JP2 header box to extract size, component count and
color space information, returning a (size, mode, mimetype) tuple."""
# Find the JP2 header box
header = None
mimetype = None
while True:
2019-03-21 16:28:20 +03:00
lbox, tbox = struct.unpack(">I4s", fp.read(8))
if lbox == 1:
2019-03-21 16:28:20 +03:00
lbox = struct.unpack(">Q", fp.read(8))[0]
hlen = 16
else:
hlen = 8
if lbox < hlen:
2019-03-21 16:28:20 +03:00
raise SyntaxError("Invalid JP2 header length")
2014-08-26 17:47:10 +04:00
2019-03-21 16:28:20 +03:00
if tbox == b"jp2h":
header = fp.read(lbox - hlen)
break
2019-03-21 16:28:20 +03:00
elif tbox == b"ftyp":
if fp.read(4) == b"jpx ":
mimetype = "image/jpx"
fp.seek(lbox - hlen - 4, os.SEEK_CUR)
else:
fp.seek(lbox - hlen, os.SEEK_CUR)
if header is None:
2019-03-21 16:28:20 +03:00
raise SyntaxError("could not find JP2 header")
size = None
mode = None
bpc = None
nc = None
2016-09-03 05:23:42 +03:00
hio = io.BytesIO(header)
while True:
2019-03-21 16:28:20 +03:00
lbox, tbox = struct.unpack(">I4s", hio.read(8))
if lbox == 1:
2019-03-21 16:28:20 +03:00
lbox = struct.unpack(">Q", hio.read(8))[0]
hlen = 16
else:
hlen = 8
content = hio.read(lbox - hlen)
2019-03-21 16:28:20 +03:00
if tbox == b"ihdr":
height, width, nc, bpc, c, unkc, ipr = struct.unpack(">IIHBBBB", content)
size = (width, height)
if unkc:
2019-03-21 16:28:20 +03:00
if nc == 1 and (bpc & 0x7F) > 8:
mode = "I;16"
elif nc == 1:
2019-03-21 16:28:20 +03:00
mode = "L"
elif nc == 2:
2019-03-21 16:28:20 +03:00
mode = "LA"
elif nc == 3:
2019-03-21 16:28:20 +03:00
mode = "RGB"
elif nc == 4:
2019-03-21 16:28:20 +03:00
mode = "RGBA"
break
2019-03-21 16:28:20 +03:00
elif tbox == b"colr":
meth, prec, approx = struct.unpack_from(">BBB", content)
if meth == 1:
2019-03-21 16:28:20 +03:00
cs = struct.unpack_from(">I", content, 3)[0]
if cs == 16: # sRGB
if nc == 1 and (bpc & 0x7F) > 8:
mode = "I;16"
elif nc == 1:
2019-03-21 16:28:20 +03:00
mode = "L"
elif nc == 3:
2019-03-21 16:28:20 +03:00
mode = "RGB"
elif nc == 4:
2019-03-21 16:28:20 +03:00
mode = "RGBA"
break
elif cs == 17: # grayscale
2019-03-21 16:28:20 +03:00
if nc == 1 and (bpc & 0x7F) > 8:
mode = "I;16"
elif nc == 1:
2019-03-21 16:28:20 +03:00
mode = "L"
elif nc == 2:
2019-03-21 16:28:20 +03:00
mode = "LA"
break
elif cs == 18: # sYCC
if nc == 3:
2019-03-21 16:28:20 +03:00
mode = "RGB"
elif nc == 4:
2019-03-21 16:28:20 +03:00
mode = "RGBA"
break
if size is None or mode is None:
raise SyntaxError("Malformed jp2 header")
2016-09-03 05:23:42 +03:00
return (size, mode, mimetype)
2019-03-21 16:28:20 +03:00
##
# Image plugin for JPEG2000 images.
class Jpeg2KImageFile(ImageFile.ImageFile):
format = "JPEG2000"
format_description = "JPEG 2000 (ISO 15444)"
def _open(self):
sig = self.fp.read(4)
2019-03-21 16:28:20 +03:00
if sig == b"\xff\x4f\xff\x51":
self.codec = "j2k"
self._size, self.mode = _parse_codestream(self.fp)
else:
sig = sig + self.fp.read(8)
2019-03-21 16:28:20 +03:00
if sig == b"\x00\x00\x00\x0cjP \x0d\x0a\x87\x0a":
self.codec = "jp2"
header = _parse_jp2_header(self.fp)
self._size, self.mode, self.custom_mimetype = header
else:
2019-03-21 16:28:20 +03:00
raise SyntaxError("not a JPEG 2000 file")
if self.size is None or self.mode is None:
2019-03-21 16:28:20 +03:00
raise SyntaxError("unable to determine size/mode")
2020-03-24 11:19:46 +03:00
self._reduce = 0
self.layers = 0
fd = -1
2014-05-27 15:43:54 +04:00
length = -1
try:
fd = self.fp.fileno()
length = os.fstat(fd).st_size
except Exception:
fd = -1
try:
pos = self.fp.tell()
self.fp.seek(0, io.SEEK_END)
length = self.fp.tell()
self.fp.seek(pos)
except Exception:
2014-05-27 15:43:54 +04:00
length = -1
2019-03-21 16:28:20 +03:00
self.tile = [
(
"jpeg2k",
(0, 0) + self.size,
0,
2020-03-24 11:19:46 +03:00
(self.codec, self._reduce, self.layers, fd, length),
2019-03-21 16:28:20 +03:00
)
]
2020-03-24 11:19:46 +03:00
@property
def reduce(self):
2020-03-20 09:49:36 +03:00
# https://github.com/python-pillow/Pillow/issues/4343 found that the
# new Image 'reduce' method was shadowed by this plugin's 'reduce'
# property. This attempts to allow for both scenarios
2020-03-24 11:19:46 +03:00
return self._reduce or super().reduce
@reduce.setter
def reduce(self, value):
self._reduce = value
def load(self):
if self.tile and self._reduce:
2020-03-24 11:19:46 +03:00
power = 1 << self._reduce
2014-03-13 15:57:47 +04:00
adjust = power >> 1
2019-03-21 16:28:20 +03:00
self._size = (
int((self.size[0] + adjust) / power),
int((self.size[1] + adjust) / power),
)
# Update the reduce and layers settings
t = self.tile[0]
2020-03-24 11:19:46 +03:00
t3 = (t[3][0], self._reduce, self.layers, t[3][3], t[3][4])
self.tile = [(t[0], (0, 0) + self.size, t[2], t3)]
return ImageFile.ImageFile.load(self)
def _accept(prefix):
2019-03-21 16:28:20 +03:00
return (
prefix[:4] == b"\xff\x4f\xff\x51"
or prefix[:12] == b"\x00\x00\x00\x0cjP \x0d\x0a\x87\x0a"
)
2014-03-13 22:27:16 +04:00
# ------------------------------------------------------------
# Save support
2019-03-21 16:28:20 +03:00
2014-03-13 22:27:16 +04:00
def _save(im, fp, filename):
2019-03-21 16:28:20 +03:00
if filename.endswith(".j2k"):
kind = "j2k"
2014-03-13 22:27:16 +04:00
else:
2019-03-21 16:28:20 +03:00
kind = "jp2"
# Get the keyword arguments
info = im.encoderinfo
2019-03-21 16:28:20 +03:00
offset = info.get("offset", None)
tile_offset = info.get("tile_offset", None)
tile_size = info.get("tile_size", None)
quality_mode = info.get("quality_mode", "rates")
quality_layers = info.get("quality_layers", None)
2018-11-16 15:31:42 +03:00
if quality_layers is not None and not (
2019-03-21 16:28:20 +03:00
isinstance(quality_layers, (list, tuple))
and all(
[
isinstance(quality_layer, (int, float))
for quality_layer in quality_layers
]
)
2018-11-16 15:31:42 +03:00
):
2019-03-21 16:28:20 +03:00
raise ValueError("quality_layers must be a sequence of numbers")
num_resolutions = info.get("num_resolutions", 0)
cblk_size = info.get("codeblock_size", None)
precinct_size = info.get("precinct_size", None)
irreversible = info.get("irreversible", False)
progression = info.get("progression", "LRCP")
cinema_mode = info.get("cinema_mode", "no")
fd = -1
if hasattr(fp, "fileno"):
try:
fd = fp.fileno()
except Exception:
fd = -1
im.encoderconfig = (
offset,
tile_offset,
tile_size,
quality_mode,
quality_layers,
num_resolutions,
cblk_size,
precinct_size,
irreversible,
progression,
cinema_mode,
2019-03-21 16:28:20 +03:00
fd,
)
2019-03-21 16:28:20 +03:00
ImageFile._save(im, fp, [("jpeg2k", (0, 0) + im.size, 0, kind)])
# ------------------------------------------------------------
# Registry stuff
2018-03-03 12:54:00 +03:00
Image.register_open(Jpeg2KImageFile.format, Jpeg2KImageFile, _accept)
Image.register_save(Jpeg2KImageFile.format, _save)
2019-03-21 16:28:20 +03:00
Image.register_extensions(
Jpeg2KImageFile.format, [".jp2", ".j2k", ".jpc", ".jpf", ".jpx", ".j2c"]
)
2019-03-21 16:28:20 +03:00
Image.register_mime(Jpeg2KImageFile.format, "image/jp2")