Jpeg2KImagePlugin.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362
  1. #
  2. # The Python Imaging Library
  3. # $Id$
  4. #
  5. # JPEG2000 file handling
  6. #
  7. # History:
  8. # 2014-03-12 ajh Created
  9. # 2021-06-30 rogermb Extract dpi information from the 'resc' header box
  10. #
  11. # Copyright (c) 2014 Coriolis Systems Limited
  12. # Copyright (c) 2014 Alastair Houghton
  13. #
  14. # See the README file for information on usage and redistribution.
  15. #
  16. import io
  17. import os
  18. import struct
  19. from . import Image, ImageFile
  20. class BoxReader:
  21. """
  22. A small helper class to read fields stored in JPEG2000 header boxes
  23. and to easily step into and read sub-boxes.
  24. """
  25. def __init__(self, fp, length=-1):
  26. self.fp = fp
  27. self.has_length = length >= 0
  28. self.length = length
  29. self.remaining_in_box = -1
  30. def _can_read(self, num_bytes):
  31. if self.has_length and self.fp.tell() + num_bytes > self.length:
  32. # Outside box: ensure we don't read past the known file length
  33. return False
  34. if self.remaining_in_box >= 0:
  35. # Inside box contents: ensure read does not go past box boundaries
  36. return num_bytes <= self.remaining_in_box
  37. else:
  38. return True # No length known, just read
  39. def _read_bytes(self, num_bytes):
  40. if not self._can_read(num_bytes):
  41. raise SyntaxError("Not enough data in header")
  42. data = self.fp.read(num_bytes)
  43. if len(data) < num_bytes:
  44. raise OSError(
  45. f"Expected to read {num_bytes} bytes but only got {len(data)}."
  46. )
  47. if self.remaining_in_box > 0:
  48. self.remaining_in_box -= num_bytes
  49. return data
  50. def read_fields(self, field_format):
  51. size = struct.calcsize(field_format)
  52. data = self._read_bytes(size)
  53. return struct.unpack(field_format, data)
  54. def read_boxes(self):
  55. size = self.remaining_in_box
  56. data = self._read_bytes(size)
  57. return BoxReader(io.BytesIO(data), size)
  58. def has_next_box(self):
  59. if self.has_length:
  60. return self.fp.tell() + self.remaining_in_box < self.length
  61. else:
  62. return True
  63. def next_box_type(self):
  64. # Skip the rest of the box if it has not been read
  65. if self.remaining_in_box > 0:
  66. self.fp.seek(self.remaining_in_box, os.SEEK_CUR)
  67. self.remaining_in_box = -1
  68. # Read the length and type of the next box
  69. lbox, tbox = self.read_fields(">I4s")
  70. if lbox == 1:
  71. lbox = self.read_fields(">Q")[0]
  72. hlen = 16
  73. else:
  74. hlen = 8
  75. if lbox < hlen or not self._can_read(lbox - hlen):
  76. raise SyntaxError("Invalid header length")
  77. self.remaining_in_box = lbox - hlen
  78. return tbox
  79. def _parse_codestream(fp):
  80. """Parse the JPEG 2000 codestream to extract the size and component
  81. count from the SIZ marker segment, returning a PIL (size, mode) tuple."""
  82. hdr = fp.read(2)
  83. lsiz = struct.unpack(">H", hdr)[0]
  84. siz = hdr + fp.read(lsiz - 2)
  85. lsiz, rsiz, xsiz, ysiz, xosiz, yosiz, _, _, _, _, csiz = struct.unpack_from(
  86. ">HHIIIIIIIIH", siz
  87. )
  88. ssiz = [None] * csiz
  89. xrsiz = [None] * csiz
  90. yrsiz = [None] * csiz
  91. for i in range(csiz):
  92. ssiz[i], xrsiz[i], yrsiz[i] = struct.unpack_from(">BBB", siz, 36 + 3 * i)
  93. size = (xsiz - xosiz, ysiz - yosiz)
  94. if csiz == 1:
  95. if (yrsiz[0] & 0x7F) > 8:
  96. mode = "I;16"
  97. else:
  98. mode = "L"
  99. elif csiz == 2:
  100. mode = "LA"
  101. elif csiz == 3:
  102. mode = "RGB"
  103. elif csiz == 4:
  104. mode = "RGBA"
  105. else:
  106. mode = None
  107. return size, mode
  108. def _res_to_dpi(num, denom, exp):
  109. """Convert JPEG2000's (numerator, denominator, exponent-base-10) resolution,
  110. calculated as (num / denom) * 10^exp and stored in dots per meter,
  111. to floating-point dots per inch."""
  112. if denom != 0:
  113. return (254 * num * (10**exp)) / (10000 * denom)
  114. def _parse_jp2_header(fp):
  115. """Parse the JP2 header box to extract size, component count,
  116. color space information, and optionally DPI information,
  117. returning a (size, mode, mimetype, dpi) tuple."""
  118. # Find the JP2 header box
  119. reader = BoxReader(fp)
  120. header = None
  121. mimetype = None
  122. while reader.has_next_box():
  123. tbox = reader.next_box_type()
  124. if tbox == b"jp2h":
  125. header = reader.read_boxes()
  126. break
  127. elif tbox == b"ftyp":
  128. if reader.read_fields(">4s")[0] == b"jpx ":
  129. mimetype = "image/jpx"
  130. size = None
  131. mode = None
  132. bpc = None
  133. nc = None
  134. dpi = None # 2-tuple of DPI info, or None
  135. while header.has_next_box():
  136. tbox = header.next_box_type()
  137. if tbox == b"ihdr":
  138. height, width, nc, bpc = header.read_fields(">IIHB")
  139. size = (width, height)
  140. if nc == 1 and (bpc & 0x7F) > 8:
  141. mode = "I;16"
  142. elif nc == 1:
  143. mode = "L"
  144. elif nc == 2:
  145. mode = "LA"
  146. elif nc == 3:
  147. mode = "RGB"
  148. elif nc == 4:
  149. mode = "RGBA"
  150. elif tbox == b"res ":
  151. res = header.read_boxes()
  152. while res.has_next_box():
  153. tres = res.next_box_type()
  154. if tres == b"resc":
  155. vrcn, vrcd, hrcn, hrcd, vrce, hrce = res.read_fields(">HHHHBB")
  156. hres = _res_to_dpi(hrcn, hrcd, hrce)
  157. vres = _res_to_dpi(vrcn, vrcd, vrce)
  158. if hres is not None and vres is not None:
  159. dpi = (hres, vres)
  160. break
  161. if size is None or mode is None:
  162. raise SyntaxError("Malformed JP2 header")
  163. return size, mode, mimetype, dpi
  164. ##
  165. # Image plugin for JPEG2000 images.
  166. class Jpeg2KImageFile(ImageFile.ImageFile):
  167. format = "JPEG2000"
  168. format_description = "JPEG 2000 (ISO 15444)"
  169. def _open(self):
  170. sig = self.fp.read(4)
  171. if sig == b"\xff\x4f\xff\x51":
  172. self.codec = "j2k"
  173. self._size, self.mode = _parse_codestream(self.fp)
  174. else:
  175. sig = sig + self.fp.read(8)
  176. if sig == b"\x00\x00\x00\x0cjP \x0d\x0a\x87\x0a":
  177. self.codec = "jp2"
  178. header = _parse_jp2_header(self.fp)
  179. self._size, self.mode, self.custom_mimetype, dpi = header
  180. if dpi is not None:
  181. self.info["dpi"] = dpi
  182. else:
  183. raise SyntaxError("not a JPEG 2000 file")
  184. if self.size is None or self.mode is None:
  185. raise SyntaxError("unable to determine size/mode")
  186. self._reduce = 0
  187. self.layers = 0
  188. fd = -1
  189. length = -1
  190. try:
  191. fd = self.fp.fileno()
  192. length = os.fstat(fd).st_size
  193. except Exception:
  194. fd = -1
  195. try:
  196. pos = self.fp.tell()
  197. self.fp.seek(0, io.SEEK_END)
  198. length = self.fp.tell()
  199. self.fp.seek(pos)
  200. except Exception:
  201. length = -1
  202. self.tile = [
  203. (
  204. "jpeg2k",
  205. (0, 0) + self.size,
  206. 0,
  207. (self.codec, self._reduce, self.layers, fd, length),
  208. )
  209. ]
  210. @property
  211. def reduce(self):
  212. # https://github.com/python-pillow/Pillow/issues/4343 found that the
  213. # new Image 'reduce' method was shadowed by this plugin's 'reduce'
  214. # property. This attempts to allow for both scenarios
  215. return self._reduce or super().reduce
  216. @reduce.setter
  217. def reduce(self, value):
  218. self._reduce = value
  219. def load(self):
  220. if self.tile and self._reduce:
  221. power = 1 << self._reduce
  222. adjust = power >> 1
  223. self._size = (
  224. int((self.size[0] + adjust) / power),
  225. int((self.size[1] + adjust) / power),
  226. )
  227. # Update the reduce and layers settings
  228. t = self.tile[0]
  229. t3 = (t[3][0], self._reduce, self.layers, t[3][3], t[3][4])
  230. self.tile = [(t[0], (0, 0) + self.size, t[2], t3)]
  231. return ImageFile.ImageFile.load(self)
  232. def _accept(prefix):
  233. return (
  234. prefix[:4] == b"\xff\x4f\xff\x51"
  235. or prefix[:12] == b"\x00\x00\x00\x0cjP \x0d\x0a\x87\x0a"
  236. )
  237. # ------------------------------------------------------------
  238. # Save support
  239. def _save(im, fp, filename):
  240. # Get the keyword arguments
  241. info = im.encoderinfo
  242. if filename.endswith(".j2k") or info.get("no_jp2", False):
  243. kind = "j2k"
  244. else:
  245. kind = "jp2"
  246. offset = info.get("offset", None)
  247. tile_offset = info.get("tile_offset", None)
  248. tile_size = info.get("tile_size", None)
  249. quality_mode = info.get("quality_mode", "rates")
  250. quality_layers = info.get("quality_layers", None)
  251. if quality_layers is not None and not (
  252. isinstance(quality_layers, (list, tuple))
  253. and all(
  254. [
  255. isinstance(quality_layer, (int, float))
  256. for quality_layer in quality_layers
  257. ]
  258. )
  259. ):
  260. raise ValueError("quality_layers must be a sequence of numbers")
  261. num_resolutions = info.get("num_resolutions", 0)
  262. cblk_size = info.get("codeblock_size", None)
  263. precinct_size = info.get("precinct_size", None)
  264. irreversible = info.get("irreversible", False)
  265. progression = info.get("progression", "LRCP")
  266. cinema_mode = info.get("cinema_mode", "no")
  267. mct = info.get("mct", 0)
  268. fd = -1
  269. if hasattr(fp, "fileno"):
  270. try:
  271. fd = fp.fileno()
  272. except Exception:
  273. fd = -1
  274. im.encoderconfig = (
  275. offset,
  276. tile_offset,
  277. tile_size,
  278. quality_mode,
  279. quality_layers,
  280. num_resolutions,
  281. cblk_size,
  282. precinct_size,
  283. irreversible,
  284. progression,
  285. cinema_mode,
  286. mct,
  287. fd,
  288. )
  289. ImageFile._save(im, fp, [("jpeg2k", (0, 0) + im.size, 0, kind)])
  290. # ------------------------------------------------------------
  291. # Registry stuff
  292. Image.register_open(Jpeg2KImageFile.format, Jpeg2KImageFile, _accept)
  293. Image.register_save(Jpeg2KImageFile.format, _save)
  294. Image.register_extensions(
  295. Jpeg2KImageFile.format, [".jp2", ".j2k", ".jpc", ".jpf", ".jpx", ".j2c"]
  296. )
  297. Image.register_mime(Jpeg2KImageFile.format, "image/jp2")