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>
|
|
|
|
#
|
2016-09-17 11:03:40 +03:00
|
|
|
#
|
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-09-17 11:03:40 +03:00
|
|
|
# 2016-16-10 mb Add save method without compression
|
2010-07-31 06:52:47 +04:00
|
|
|
# 1995-09-10 fl Created
|
|
|
|
#
|
2016-09-17 11:03:40 +03:00
|
|
|
# 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.
|
|
|
|
#
|
|
|
|
|
|
|
|
|
2017-01-17 16:22:18 +03:00
|
|
|
from . import Image, ImageFile
|
2017-07-27 00:01:45 +03:00
|
|
|
from ._binary import i8, o8, i16be as i16, o16be as o16
|
2016-09-17 11:03:40 +03:00
|
|
|
import struct
|
|
|
|
import os
|
2017-07-29 00:30:55 +03:00
|
|
|
import sys
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2017-07-27 00:01:45 +03:00
|
|
|
|
2016-09-17 11:03:40 +03:00
|
|
|
__version__ = "0.3"
|
2015-08-25 15:27:18 +03:00
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
|
|
|
|
def _accept(prefix):
|
2015-06-18 03:12:12 +03:00
|
|
|
return len(prefix) >= 2 and i16(prefix) == 474
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2014-07-16 22:30:41 +04:00
|
|
|
|
2017-07-23 21:41:48 +03: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"
|
|
|
|
|
|
|
|
def _open(self):
|
|
|
|
|
|
|
|
# HEAD
|
2017-07-22 18:00:15 +03:00
|
|
|
headlen = 512
|
|
|
|
s = self.fp.read(headlen)
|
2017-07-22 10:34:06 +03:00
|
|
|
|
|
|
|
# magic number : 474
|
2010-07-31 06:52:47 +04:00
|
|
|
if i16(s) != 474:
|
2014-07-20 02:50:05 +04:00
|
|
|
raise ValueError("Not an SGI image file")
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2017-07-22 10:34:06 +03:00
|
|
|
# compression : verbatim or RLE
|
py3k: The big push
There are two main issues fixed with this commit:
* bytes vs. str: All file, image, and palette data are now handled as
bytes. A new _binary module consolidates the hacks needed to do this
across Python versions. tostring/fromstring methods have been renamed to
tobytes/frombytes, but the Python 2.6/2.7 versions alias them to the old
names for compatibility. Users should move to tobytes/frombytes.
One other potentially-breaking change is that text data in image files
(such as tags, comments) are now explicitly handled with a specific
character encoding in mind. This works well with the Unicode str in
Python 3, but may trip up old code expecting a straight byte-for-byte
translation to a Python string. This also required a change to Gohlke's
tags tests (in Tests/test_file_png.py) to expect Unicode strings from
the code.
* True div vs. floor div: Many division operations used the "/" operator
to do floor division, which is now the "//" operator in Python 3. These
were fixed.
As of this commit, on the first pass, I have one failing test (improper
handling of a slice object in a C module, test_imagepath.py) in Python 3,
and three that that I haven't tried running yet (test_imagegl,
test_imagegrab, and test_imageqt). I also haven't tested anything on
Windows. All but the three skipped tests run flawlessly against Pythons
2.6 and 2.7.
2012-10-21 01:01:53 +04:00
|
|
|
compression = i8(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 = i8(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:])
|
|
|
|
|
|
|
|
# xsize : width
|
|
|
|
xsize = i16(s[6:])
|
|
|
|
|
|
|
|
# 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:])
|
|
|
|
|
|
|
|
# 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
|
2017-07-23 21:41:48 +03:00
|
|
|
rawmode = ""
|
|
|
|
try:
|
|
|
|
rawmode = MODES[layout]
|
|
|
|
except KeyError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
if rawmode == "":
|
2014-07-20 02:50:05 +04:00
|
|
|
raise ValueError("Unsupported SGI image mode")
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2017-07-22 10:34:06 +03:00
|
|
|
self.size = xsize, ysize
|
2017-07-23 21:41:48 +03:00
|
|
|
self.mode = rawmode.split(";")[0]
|
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
|
2017-07-23 21:41:48 +03:00
|
|
|
if bpc == 2:
|
|
|
|
self.tile = [("SGI16", (0, 0) + self.size,
|
|
|
|
headlen, (self.mode, 0, orientation))]
|
|
|
|
else:
|
|
|
|
self.tile = []
|
|
|
|
offset = headlen
|
|
|
|
for layer in self.mode:
|
|
|
|
self.tile.append(
|
|
|
|
("raw", (0, 0) + self.size,
|
|
|
|
offset, (layer, 0, orientation)))
|
|
|
|
offset += pagesize
|
2010-07-31 06:52:47 +04:00
|
|
|
elif compression == 1:
|
2017-07-22 10:34:06 +03:00
|
|
|
self.tile = [("sgi_rle", (0, 0) + self.size,
|
2017-07-25 08:19:20 +03:00
|
|
|
headlen, (rawmode, orientation, bpc))]
|
2010-07-31 06:52:47 +04:00
|
|
|
|
2016-09-17 11:03:40 +03:00
|
|
|
|
|
|
|
def _save(im, fp, filename):
|
|
|
|
if im.mode != "RGB" and im.mode != "RGBA" and im.mode != "L":
|
|
|
|
raise ValueError("Unsupported SGI image mode")
|
|
|
|
|
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 != 1 and bpc != 2:
|
|
|
|
raise ValueError("Unsupported number of bytes per pixel")
|
|
|
|
|
2016-09-17 11:03:40 +03:00
|
|
|
# Flip the image, since the origin of SGI file is the bottom-left corner
|
|
|
|
im = im.transpose(Image.FLIP_TOP_BOTTOM)
|
|
|
|
# Define the file as SGI File Format
|
|
|
|
magicNumber = 474
|
|
|
|
# Run-Length Encoding Compression - Unsupported at this time
|
|
|
|
rle = 0
|
2017-07-27 00:01:45 +03:00
|
|
|
|
2016-09-17 11:03:40 +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
|
|
|
|
2016-09-17 11:03:40 +03:00
|
|
|
if dim == 1 or dim == 2:
|
|
|
|
z = 1
|
2017-09-29 12:41:26 +03:00
|
|
|
|
2017-07-28 12:51:26 +03:00
|
|
|
# assert we've got the right number of bands.
|
|
|
|
if len(im.getbands()) != z:
|
|
|
|
raise ValueError("incorrect number of bands in SGI write: %s vs %s" %
|
|
|
|
(z, len(im.getbands())))
|
|
|
|
|
2016-09-17 11:03:40 +03:00
|
|
|
# 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)
|
|
|
|
imgName = os.path.splitext(os.path.basename(filename))[0]
|
|
|
|
if str is not bytes:
|
|
|
|
imgName = imgName.encode('ascii', 'ignore')
|
2016-09-17 11:03:40 +03:00
|
|
|
# Standard representation of pixel in the file
|
|
|
|
colormap = 0
|
|
|
|
fp.write(struct.pack('>h', magicNumber))
|
2016-12-31 02:10:47 +03:00
|
|
|
fp.write(o8(rle))
|
|
|
|
fp.write(o8(bpc))
|
2016-09-17 11:03:40 +03:00
|
|
|
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', imgName)) # truncates to 79 chars
|
|
|
|
fp.write(struct.pack('s', b'')) # force null byte after imgname
|
2016-09-17 11:03:40 +03:00
|
|
|
fp.write(struct.pack('>l', colormap))
|
2017-04-20 14:14:23 +03:00
|
|
|
fp.write(struct.pack('404s', b'')) # dummy
|
2016-12-31 02:10:47 +03:00
|
|
|
|
|
|
|
for channel in im.split():
|
2017-07-27 00:01:45 +03:00
|
|
|
rawchannel = channel.tobytes()
|
|
|
|
if bpc == 1:
|
|
|
|
fp.write(rawchannel)
|
|
|
|
else:
|
|
|
|
for pixel in rawchannel:
|
|
|
|
fp.write(o16(i8(pixel) * 256))
|
2016-12-31 02:10:47 +03:00
|
|
|
|
2016-09-17 11:03:40 +03:00
|
|
|
fp.close()
|
|
|
|
|
|
|
|
|
2017-07-23 21:41:48 +03:00
|
|
|
class SGI16Decoder(ImageFile.PyDecoder):
|
2017-07-27 00:01:45 +03:00
|
|
|
_pulls_fd = True
|
2017-07-23 21:41:48 +03:00
|
|
|
|
|
|
|
def decode(self, buffer):
|
2017-07-25 08:19:20 +03:00
|
|
|
rawmode, stride, orientation = self.args
|
2017-07-23 21:41:48 +03:00
|
|
|
pagesize = self.state.xsize * self.state.ysize
|
|
|
|
zsize = len(self.mode)
|
|
|
|
data = bytearray(pagesize * zsize)
|
2017-07-27 00:01:45 +03:00
|
|
|
self.fd.seek(512)
|
|
|
|
s = self.fd.read(2 * pagesize * zsize)
|
2017-07-23 21:41:48 +03:00
|
|
|
i = 0
|
2017-07-25 08:19:20 +03:00
|
|
|
y = 0
|
|
|
|
if orientation < 0:
|
|
|
|
y = self.state.ysize - 1
|
|
|
|
while y >= 0 and y < self.state.ysize:
|
2017-07-23 21:41:48 +03:00
|
|
|
for x in range(self.state.xsize):
|
|
|
|
for z in range(zsize):
|
2017-07-25 08:19:20 +03:00
|
|
|
bi = (x + y * self.state.xsize +
|
|
|
|
y * stride + z * pagesize) * 2
|
2017-07-27 00:01:45 +03:00
|
|
|
pixel = i16(s, o=bi)
|
2017-07-23 21:41:48 +03:00
|
|
|
pixel = int(pixel // 256)
|
2017-07-29 00:30:55 +03:00
|
|
|
if sys.version_info.major == 3:
|
|
|
|
data[i] = pixel
|
|
|
|
else:
|
|
|
|
data[i] = o8(pixel)
|
2017-07-23 21:41:48 +03:00
|
|
|
i += 1
|
2017-07-25 08:19:20 +03:00
|
|
|
y += orientation
|
2017-07-23 21:41:48 +03:00
|
|
|
self.set_as_raw(bytes(data))
|
|
|
|
return -1, 0
|
|
|
|
|
2010-07-31 06:52:47 +04:00
|
|
|
#
|
|
|
|
# registry
|
|
|
|
|
2017-07-23 21:41:48 +03:00
|
|
|
|
|
|
|
Image.register_decoder("SGI16", SGI16Decoder)
|
2015-07-04 16:29:58 +03:00
|
|
|
Image.register_open(SgiImageFile.format, SgiImageFile, _accept)
|
2016-09-17 11:03:40 +03:00
|
|
|
Image.register_save(SgiImageFile.format, _save)
|
|
|
|
Image.register_mime(SgiImageFile.format, "image/sgi")
|
|
|
|
Image.register_mime(SgiImageFile.format, "image/rgb")
|
2016-04-25 07:59:02 +03:00
|
|
|
|
|
|
|
Image.register_extensions(SgiImageFile.format, [".bw", ".rgb", ".rgba", ".sgi"])
|
2016-09-17 11:03:40 +03:00
|
|
|
|
|
|
|
# End of file
|