# # The Python Imaging Library. # $Id$ # # SGI image file handling # # See "The SGI Image File Format (Draft version 0.97)", Paul Haeberli. # # # # History: # 2017-22-07 mb Add RLE decompression # 2016-16-10 mb Add save method without compression # 1995-09-10 fl Created # # Copyright (c) 2016 by Mickael Bonfill. # 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 . import Image, ImageFile from ._binary import i8, o8, i16be as i16, o16be as o16 import struct import os import sys __version__ = "0.3" def _accept(prefix): return len(prefix) >= 2 and i16(prefix) == 474 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" } ## # Image plugin for SGI images. class SgiImageFile(ImageFile.ImageFile): format = "SGI" format_description = "SGI Image File Format" def _open(self): # HEAD headlen = 512 s = self.fp.read(headlen) # magic number : 474 if i16(s) != 474: raise ValueError("Not an SGI image file") # compression : verbatim or RLE compression = i8(s[2]) # bpc : 1 or 2 bytes (8bits or 16bits) bpc = i8(s[3]) # 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:]) # zsize : channels count zsize = i16(s[10:]) # layout layout = bpc, dimension, zsize # determine mode from bits/zsize rawmode = "" try: rawmode = MODES[layout] except KeyError: pass if rawmode == "": raise ValueError("Unsupported SGI image mode") self.size = xsize, ysize self.mode = rawmode.split(";")[0] # orientation -1 : scanlines begins at the bottom-left corner orientation = -1 # decoder info if compression == 0: pagesize = xsize * ysize * bpc 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 elif compression == 1: self.tile = [("sgi_rle", (0, 0) + self.size, headlen, (rawmode, orientation, bpc))] def _save(im, fp, filename): if im.mode != "RGB" and im.mode != "RGBA" and im.mode != "L": raise ValueError("Unsupported SGI image mode") # 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") # 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 # 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) if dim == 1 or dim == 2: z = 1 # 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()))) # Minimum Byte value pinmin = 0 # Maximum Byte value (255 = 8bits per pixel) pinmax = 255 # 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') # Standard representation of pixel in the file colormap = 0 fp.write(struct.pack('>h', magicNumber)) 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)) 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 fp.write(struct.pack('>l', colormap)) fp.write(struct.pack('404s', b'')) # dummy for channel in im.split(): rawchannel = channel.tobytes() if bpc == 1: fp.write(rawchannel) else: for pixel in rawchannel: fp.write(o16(i8(pixel) * 256)) fp.close() class SGI16Decoder(ImageFile.PyDecoder): _pulls_fd = True def decode(self, buffer): rawmode, stride, orientation = self.args pagesize = self.state.xsize * self.state.ysize zsize = len(self.mode) data = bytearray(pagesize * zsize) self.fd.seek(512) s = self.fd.read(2 * pagesize * zsize) print(len(s)) i = 0 y = 0 if orientation < 0: y = self.state.ysize - 1 while y >= 0 and y < self.state.ysize: for x in range(self.state.xsize): for z in range(zsize): bi = (x + y * self.state.xsize + y * stride + z * pagesize) * 2 pixel = i16(s, o=bi) pixel = int(pixel // 256) if sys.version_info.major == 3: data[i] = pixel else: data[i] = o8(pixel) i += 1 y += orientation self.set_as_raw(bytes(data)) return -1, 0 # # 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") Image.register_mime(SgiImageFile.format, "image/rgb") Image.register_extensions(SgiImageFile.format, [".bw", ".rgb", ".rgba", ".sgi"]) # End of file