Pillow/src/PIL/SgiImagePlugin.py

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

248 lines
6.6 KiB
Python
Raw Normal View History

2010-07-31 06:52:47 +04:00
#
# The Python Imaging Library.
# $Id$
#
# SGI image file handling
#
# See "The SGI Image File Format (Draft version 0.97)", Paul Haeberli.
# <ftp://ftp.sgi.com/graphics/SGIIMAGESPEC>
#
#
2010-07-31 06:52:47 +04:00
# History:
2017-07-22 10:34:06 +03:00
# 2017-22-07 mb Add RLE decompression
# 2016-16-10 mb Add save method without compression
2010-07-31 06:52:47 +04:00
# 1995-09-10 fl Created
#
# Copyright (c) 2016 by Mickael Bonfill.
2010-07-31 06:52:47 +04:00
# Copyright (c) 2008 by Karsten Hiddemann.
# Copyright (c) 1997 by Secret Labs AB.
# Copyright (c) 1995 by Fredrik Lundh.
#
# See the README file for information on usage and redistribution.
#
from __future__ import annotations
2010-07-31 06:52:47 +04:00
import os
import struct
2024-02-13 14:26:23 +03:00
from typing import IO
2010-07-31 06:52:47 +04:00
from . import Image, ImageFile
2020-09-01 20:16:46 +03:00
from ._binary import i16be as i16
from ._binary import o8
2017-07-27 00:01:45 +03:00
2010-07-31 06:52:47 +04:00
2024-01-16 04:22:59 +03:00
def _accept(prefix: bytes) -> bool:
return len(prefix) >= 2 and i16(prefix) == 474
2010-07-31 06:52:47 +04:00
2014-07-16 22:30:41 +04:00
MODES = {
(1, 1, 1): "L",
(1, 2, 1): "L",
(2, 1, 1): "L;16B",
(2, 2, 1): "L;16B",
(1, 3, 3): "RGB",
(2, 3, 3): "RGB;16B",
(1, 3, 4): "RGBA",
(2, 3, 4): "RGBA;16B",
}
2010-07-31 06:52:47 +04:00
##
# Image plugin for SGI images.
class SgiImageFile(ImageFile.ImageFile):
format = "SGI"
format_description = "SGI Image File Format"
2024-01-16 04:22:59 +03:00
def _open(self) -> None:
2010-07-31 06:52:47 +04:00
# HEAD
2024-01-16 04:22:59 +03:00
assert self.fp is not None
2017-07-22 18:00:15 +03:00
headlen = 512
s = self.fp.read(headlen)
2017-07-22 10:34:06 +03:00
if not _accept(s):
msg = "Not an SGI image file"
raise ValueError(msg)
2010-07-31 06:52:47 +04:00
2017-07-22 10:34:06 +03:00
# compression : verbatim or RLE
compression = s[2]
2010-07-31 06:52:47 +04:00
2017-07-22 18:00:15 +03:00
# bpc : 1 or 2 bytes (8bits or 16bits)
bpc = s[3]
2017-07-22 10:34:06 +03:00
# dimension : 1, 2 or 3 (depending on xsize, ysize and zsize)
dimension = i16(s, 4)
2017-07-22 10:34:06 +03:00
# xsize : width
xsize = i16(s, 6)
2017-07-22 10:34:06 +03:00
# ysize : height
ysize = i16(s, 8)
2010-07-31 06:52:47 +04:00
2017-07-22 10:34:06 +03:00
# zsize : channels count
zsize = i16(s, 10)
2017-07-22 10:34:06 +03:00
# layout
2017-07-22 18:00:15 +03:00
layout = bpc, dimension, zsize
2017-07-22 10:34:06 +03:00
# determine mode from bits/zsize
rawmode = ""
try:
rawmode = MODES[layout]
except KeyError:
pass
if rawmode == "":
msg = "Unsupported SGI image mode"
raise ValueError(msg)
2010-07-31 06:52:47 +04:00
self._size = xsize, ysize
self._mode = rawmode.split(";")[0]
if self.mode == "RGB":
self.custom_mimetype = "image/rgb"
2017-07-22 10:34:06 +03:00
# orientation -1 : scanlines begins at the bottom-left corner
orientation = -1
2010-07-31 06:52:47 +04:00
# decoder info
if compression == 0:
2017-07-22 18:00:15 +03:00
pagesize = xsize * ysize * bpc
if bpc == 2:
self.tile = [
2024-08-29 15:51:15 +03:00
ImageFile._Tile(
"SGI16",
(0, 0) + self.size,
headlen,
(self.mode, 0, orientation),
)
]
else:
self.tile = []
offset = headlen
for layer in self.mode:
self.tile.append(
2024-08-29 15:51:15 +03:00
ImageFile._Tile(
"raw", (0, 0) + self.size, offset, (layer, 0, orientation)
)
2019-03-21 16:28:20 +03:00
)
offset += pagesize
2010-07-31 06:52:47 +04:00
elif compression == 1:
2017-07-22 10:34:06 +03:00
self.tile = [
2024-08-29 15:51:15 +03:00
ImageFile._Tile(
"sgi_rle", (0, 0) + self.size, headlen, (rawmode, orientation, bpc)
)
]
2010-07-31 06:52:47 +04:00
2024-06-10 07:15:28 +03:00
def _save(im: Image.Image, fp: IO[bytes], filename: str | bytes) -> None:
if im.mode not in {"RGB", "RGBA", "L"}:
msg = "Unsupported SGI image mode"
raise ValueError(msg)
2017-07-27 00:01:45 +03:00
# Get the keyword arguments
info = im.encoderinfo
# Byte-per-pixel precision, 1 = 8bits per pixel
bpc = info.get("bpc", 1)
if bpc not in (1, 2):
msg = "Unsupported number of bytes per pixel"
raise ValueError(msg)
2017-07-27 00:01:45 +03:00
# Flip the image, since the origin of SGI file is the bottom-left corner
orientation = -1
# Define the file as SGI File Format
magic_number = 474
# Run-Length Encoding Compression - Unsupported at this time
rle = 0
2017-07-27 00:01:45 +03:00
# Number of dimensions (x,y,z)
dim = 3
# X Dimension = width / Y Dimension = height
x, y = im.size
if im.mode == "L" and y == 1:
dim = 1
elif im.mode == "L":
dim = 2
# Z Dimension: Number of channels
z = len(im.mode)
2017-07-29 09:02:14 +03:00
if dim in {1, 2}:
z = 1
2017-09-29 12:41:26 +03:00
# assert we've got the right number of bands.
if len(im.getbands()) != z:
msg = f"incorrect number of bands in SGI write: {z} vs {len(im.getbands())}"
raise ValueError(msg)
# Minimum Byte value
pinmin = 0
# Maximum Byte value (255 = 8bits per pixel)
pinmax = 255
2016-12-31 02:10:47 +03:00
# Image name (79 characters max, truncated below in write)
2024-06-10 07:15:28 +03:00
img_name = os.path.splitext(os.path.basename(filename))[0]
if isinstance(img_name, str):
img_name = img_name.encode("ascii", "ignore")
# Standard representation of pixel in the file
colormap = 0
fp.write(struct.pack(">h", magic_number))
2016-12-31 02:10:47 +03:00
fp.write(o8(rle))
fp.write(o8(bpc))
fp.write(struct.pack(">H", dim))
fp.write(struct.pack(">H", x))
fp.write(struct.pack(">H", y))
fp.write(struct.pack(">H", z))
fp.write(struct.pack(">l", pinmin))
fp.write(struct.pack(">l", pinmax))
2017-04-20 14:14:23 +03:00
fp.write(struct.pack("4s", b"")) # dummy
fp.write(struct.pack("79s", img_name)) # truncates to 79 chars
2022-04-16 15:15:21 +03:00
fp.write(struct.pack("s", b"")) # force null byte after img_name
fp.write(struct.pack(">l", colormap))
2017-04-20 14:14:23 +03:00
fp.write(struct.pack("404s", b"")) # dummy
2019-03-21 16:28:20 +03:00
rawmode = "L"
if bpc == 2:
rawmode = "L;16B"
2016-12-31 02:10:47 +03:00
for channel in im.split():
fp.write(channel.tobytes("raw", rawmode, 0, orientation))
2016-12-31 02:10:47 +03:00
2021-07-30 13:25:07 +03:00
if hasattr(fp, "flush"):
fp.flush()
2018-03-03 12:54:00 +03:00
class SGI16Decoder(ImageFile.PyDecoder):
2017-07-27 00:01:45 +03:00
_pulls_fd = True
2024-09-06 08:16:59 +03:00
def decode(self, buffer: bytes | Image.SupportsArrayInterface) -> tuple[int, int]:
2024-01-16 04:22:59 +03:00
assert self.fd is not None
assert self.im is not None
rawmode, stride, orientation = self.args
pagesize = self.state.xsize * self.state.ysize
zsize = len(self.mode)
2017-07-27 00:01:45 +03:00
self.fd.seek(512)
for band in range(zsize):
channel = Image.new("L", (self.state.xsize, self.state.ysize))
channel.frombytes(
self.fd.read(2 * pagesize), "raw", "L;16B", stride, orientation
)
self.im.putband(channel.im, band)
return -1, 0
2019-03-21 16:28:20 +03:00
2010-07-31 06:52:47 +04:00
#
# registry
Image.register_decoder("SGI16", SGI16Decoder)
Image.register_open(SgiImageFile.format, SgiImageFile, _accept)
Image.register_save(SgiImageFile.format, _save)
Image.register_mime(SgiImageFile.format, "image/sgi")
2016-04-25 07:59:02 +03:00
2018-06-24 15:32:25 +03:00
Image.register_extensions(SgiImageFile.format, [".bw", ".rgb", ".rgba", ".sgi"])
# End of file