mirror of
https://github.com/status-im/nimPNG.git
synced 2025-01-13 06:04:15 +00:00
Merge pull request #45 from jangko/fuzzing
more robust PNG loader, fuzz tested
This commit is contained in:
commit
cc16c28df8
52
.gitignore
vendored
52
.gitignore
vendored
@ -1,52 +1,2 @@
|
|||||||
# Windows image file caches
|
tests/corpus_dir
|
||||||
Thumbs.db
|
|
||||||
ehthumbs.db
|
|
||||||
|
|
||||||
# Folder config file
|
|
||||||
Desktop.ini
|
|
||||||
|
|
||||||
# Recycle Bin used on file shares
|
|
||||||
$RECYCLE.BIN/
|
|
||||||
|
|
||||||
# Windows Installer files
|
|
||||||
*.cab
|
|
||||||
*.msi
|
|
||||||
*.msm
|
|
||||||
*.msp
|
|
||||||
|
|
||||||
# Windows shortcuts
|
|
||||||
*.lnk
|
|
||||||
|
|
||||||
# =========================
|
|
||||||
# Operating System Files
|
|
||||||
# =========================
|
|
||||||
|
|
||||||
# OSX
|
|
||||||
# =========================
|
|
||||||
|
|
||||||
.DS_Store
|
|
||||||
.AppleDouble
|
|
||||||
.LSOverride
|
|
||||||
|
|
||||||
# Thumbnails
|
|
||||||
._*
|
|
||||||
|
|
||||||
# Files that might appear on external disk
|
|
||||||
.Spotlight-V100
|
|
||||||
.Trashes
|
|
||||||
|
|
||||||
# Directories potentially created on remote AFP share
|
|
||||||
.AppleDB
|
|
||||||
.AppleDesktop
|
|
||||||
Network Trash Folder
|
|
||||||
Temporary Items
|
|
||||||
.apdisk
|
|
||||||
|
|
||||||
*.exe
|
*.exe
|
||||||
nimcache
|
|
||||||
bug
|
|
||||||
tester/temp.png
|
|
||||||
tester/rainbow.png
|
|
||||||
tester/test
|
|
||||||
tester/testCodec
|
|
||||||
tester/testSuite
|
|
233
nimPNG.nim
233
nimPNG.nim
@ -25,13 +25,13 @@
|
|||||||
# part of nimPDF sister projects
|
# part of nimPDF sister projects
|
||||||
#-------------------------------------
|
#-------------------------------------
|
||||||
|
|
||||||
import streams, endians, tables, hashes, math
|
import streams, endians, tables, hashes, math, typetraits
|
||||||
import nimPNG/[buffer, nimz, filters]
|
import nimPNG/[buffer, nimz, filters]
|
||||||
|
|
||||||
import strutils
|
import strutils
|
||||||
|
|
||||||
const
|
const
|
||||||
NIM_PNG_VERSION = "0.2.6"
|
NIM_PNG_VERSION = "0.2.7"
|
||||||
|
|
||||||
type
|
type
|
||||||
PNGChunkType = distinct int32
|
PNGChunkType = distinct int32
|
||||||
@ -69,7 +69,7 @@ type
|
|||||||
PNGHeader = ref object of PNGChunk
|
PNGHeader = ref object of PNGChunk
|
||||||
width, height: int #range[1..0x7FFFFFFF]
|
width, height: int #range[1..0x7FFFFFFF]
|
||||||
bitDepth: int
|
bitDepth: int
|
||||||
colorType: PNGcolorType
|
colorType: PNGColorType
|
||||||
compressionMethod: int
|
compressionMethod: int
|
||||||
filterMethod: int
|
filterMethod: int
|
||||||
interlaceMethod: PNGInterlace
|
interlaceMethod: PNGInterlace
|
||||||
@ -181,7 +181,7 @@ type
|
|||||||
frameDataPos: int
|
frameDataPos: int
|
||||||
|
|
||||||
PNGColorMode* = ref object
|
PNGColorMode* = ref object
|
||||||
colorType*: PNGcolorType
|
colorType*: PNGColorType
|
||||||
bitDepth*: int
|
bitDepth*: int
|
||||||
paletteSize*: int
|
paletteSize*: int
|
||||||
palette*: seq[RGBA8]
|
palette*: seq[RGBA8]
|
||||||
@ -233,6 +233,8 @@ type
|
|||||||
|
|
||||||
DataBuf = Buffer[string]
|
DataBuf = Buffer[string]
|
||||||
|
|
||||||
|
PNGError* = object of CatchableError
|
||||||
|
|
||||||
proc signatureMaker(): string {. compiletime .} =
|
proc signatureMaker(): string {. compiletime .} =
|
||||||
const signatureBytes = [137, 80, 78, 71, 13, 10, 26, 10]
|
const signatureBytes = [137, 80, 78, 71, 13, 10, 26, 10]
|
||||||
result = ""
|
result = ""
|
||||||
@ -293,9 +295,8 @@ const
|
|||||||
fcTL = makeChunkType("fcTL")
|
fcTL = makeChunkType("fcTL")
|
||||||
fdAT = makeChunkType("fdAT")
|
fdAT = makeChunkType("fdAT")
|
||||||
|
|
||||||
proc PNGError(msg: string): ref Exception =
|
template PNGFatal(msg: string): untyped =
|
||||||
new(result)
|
newException(PNGError, msg)
|
||||||
result.msg = msg
|
|
||||||
|
|
||||||
proc newColorMode*(colorType=LCT_RGBA, bitDepth=8): PNGColorMode =
|
proc newColorMode*(colorType=LCT_RGBA, bitDepth=8): PNGColorMode =
|
||||||
new(result)
|
new(result)
|
||||||
@ -342,7 +343,7 @@ proc `==`(a, b: PNGColorMode): bool =
|
|||||||
proc `!=`(a, b: PNGColorMode): bool = not (a == b)
|
proc `!=`(a, b: PNGColorMode): bool = not (a == b)
|
||||||
|
|
||||||
proc readInt32(s: PNGChunk): int =
|
proc readInt32(s: PNGChunk): int =
|
||||||
if s.pos + 4 > s.data.len: raise PNGError("index out of bound 4")
|
if s.pos + 4 > s.data.len: raise PNGFatal("index out of bound 4")
|
||||||
result = ord(s.data[s.pos]) shl 8
|
result = ord(s.data[s.pos]) shl 8
|
||||||
result = (result + ord(s.data[s.pos + 1])) shl 8
|
result = (result + ord(s.data[s.pos + 1])) shl 8
|
||||||
result = (result + ord(s.data[s.pos + 2])) shl 8
|
result = (result + ord(s.data[s.pos + 2])) shl 8
|
||||||
@ -350,7 +351,7 @@ proc readInt32(s: PNGChunk): int =
|
|||||||
inc(s.pos, 4)
|
inc(s.pos, 4)
|
||||||
|
|
||||||
proc readInt16(s: PNGChunk): int =
|
proc readInt16(s: PNGChunk): int =
|
||||||
if s.pos + 2 > s.data.len: raise PNGError("index out of bound 2")
|
if s.pos + 2 > s.data.len: raise PNGFatal("index out of bound 2")
|
||||||
result = ord(s.data[s.pos]) shl 8
|
result = ord(s.data[s.pos]) shl 8
|
||||||
result = result + ord(s.data[s.pos + 1])
|
result = result + ord(s.data[s.pos + 1])
|
||||||
inc(s.pos, 2)
|
inc(s.pos, 2)
|
||||||
@ -373,12 +374,18 @@ proc readInt32BE(s: Stream): int =
|
|||||||
result = tmp
|
result = tmp
|
||||||
|
|
||||||
proc readByte(s: PNGChunk): int =
|
proc readByte(s: PNGChunk): int =
|
||||||
if s.pos + 1 > s.data.len: raise PNGError("index out of bound 1")
|
if s.pos + 1 > s.data.len: raise PNGFatal("index out of bound 1")
|
||||||
result = ord(s.data[s.pos])
|
result = ord(s.data[s.pos])
|
||||||
inc s.pos
|
inc s.pos
|
||||||
|
|
||||||
|
template readEnum(s: PNGChunk, T: type): untyped =
|
||||||
|
let typ = s.readByte.int
|
||||||
|
if typ < low(T).int or typ > high(T).int:
|
||||||
|
raise PNGFatal("Wrong " & T.name & " value " & $typ)
|
||||||
|
T(typ)
|
||||||
|
|
||||||
proc setPosition(s: PNGChunk, pos: int) =
|
proc setPosition(s: PNGChunk, pos: int) =
|
||||||
if pos < 0 or pos > s.data.len: raise PNGError("set position error")
|
if pos < 0 or pos > s.data.len: raise PNGFatal("set position error")
|
||||||
s.pos = pos
|
s.pos = pos
|
||||||
|
|
||||||
proc hasChunk*(png: PNG, chunkType: PNGChunkType): bool =
|
proc hasChunk*(png: PNG, chunkType: PNGChunkType): bool =
|
||||||
@ -399,7 +406,7 @@ proc apngGetChunk*(png: PNG, chunkType: PNGChunkType): PNGChunk =
|
|||||||
for c in png.apngChunks:
|
for c in png.apngChunks:
|
||||||
if c.chunkType == chunkType: return c
|
if c.chunkType == chunkType: return c
|
||||||
|
|
||||||
proc bitDepthAllowed(colorType: PNGcolorType, bitDepth: int): bool =
|
proc bitDepthAllowed(colorType: PNGColorType, bitDepth: int): bool =
|
||||||
case colorType
|
case colorType
|
||||||
of LCT_GREY : result = bitDepth in {1, 2, 4, 8, 16}
|
of LCT_GREY : result = bitDepth in {1, 2, 4, 8, 16}
|
||||||
of LCT_PALETTE: result = bitDepth in {1, 2, 4, 8}
|
of LCT_PALETTE: result = bitDepth in {1, 2, 4, 8}
|
||||||
@ -410,19 +417,19 @@ method parseChunk(chunk: PNGChunk, png: PNG): bool {.base, gcsafe.} = true
|
|||||||
|
|
||||||
method validateChunk(header: PNGHeader, png: PNG): bool =
|
method validateChunk(header: PNGHeader, png: PNG): bool =
|
||||||
if header.width < 1 or header.width > 0x7FFFFFFF:
|
if header.width < 1 or header.width > 0x7FFFFFFF:
|
||||||
raise PNGError("image width not allowed: " & $header.width)
|
raise PNGFatal("image width not allowed: " & $header.width)
|
||||||
if header.height < 1 or header.height > 0x7FFFFFFF:
|
if header.height < 1 or header.height > 0x7FFFFFFF:
|
||||||
raise PNGError("image width not allowed: " & $header.height)
|
raise PNGFatal("image width not allowed: " & $header.height)
|
||||||
if header.colorType notin {LCT_GREY, LCT_RGB, LCT_PALETTE, LCT_GREY_ALPHA, LCT_RGBA}:
|
if header.colorType notin {LCT_GREY, LCT_RGB, LCT_PALETTE, LCT_GREY_ALPHA, LCT_RGBA}:
|
||||||
raise PNGError("color type not allowed: " & $int(header.colorType))
|
raise PNGFatal("color type not allowed: " & $int(header.colorType))
|
||||||
if not bitDepthAllowed(header.colorType, header.bitDepth):
|
if not bitDepthAllowed(header.colorType, header.bitDepth):
|
||||||
raise PNGError("bit depth not allowed: " & $header.bitDepth)
|
raise PNGFatal("bit depth not allowed: " & $header.bitDepth)
|
||||||
if header.compressionMethod != 0:
|
if header.compressionMethod != 0:
|
||||||
raise PNGError("unsupported compression method")
|
raise PNGFatal("unsupported compression method")
|
||||||
if header.filterMethod != 0:
|
if header.filterMethod != 0:
|
||||||
raise PNGError("unsupported filter method")
|
raise PNGFatal("unsupported filter method")
|
||||||
if header.interlaceMethod notin {IM_NONE, IM_INTERLACED}:
|
if header.interlaceMethod notin {IM_NONE, IM_INTERLACED}:
|
||||||
raise PNGError("unsupported interlace method")
|
raise PNGFatal("unsupported interlace method")
|
||||||
result = true
|
result = true
|
||||||
|
|
||||||
method parseChunk(chunk: PNGHeader, png: PNG): bool =
|
method parseChunk(chunk: PNGHeader, png: PNG): bool =
|
||||||
@ -430,7 +437,7 @@ method parseChunk(chunk: PNGHeader, png: PNG): bool =
|
|||||||
chunk.width = chunk.readInt32()
|
chunk.width = chunk.readInt32()
|
||||||
chunk.height = chunk.readInt32()
|
chunk.height = chunk.readInt32()
|
||||||
chunk.bitDepth = chunk.readByte()
|
chunk.bitDepth = chunk.readByte()
|
||||||
chunk.colorType = PNGcolorType(chunk.readByte())
|
chunk.colorType = chunk.readEnum(PNGColorType)
|
||||||
chunk.compressionMethod = chunk.readByte()
|
chunk.compressionMethod = chunk.readByte()
|
||||||
chunk.filterMethod = chunk.readByte()
|
chunk.filterMethod = chunk.readByte()
|
||||||
chunk.interlaceMethod = PNGInterlace(chunk.readByte())
|
chunk.interlaceMethod = PNGInterlace(chunk.readByte())
|
||||||
@ -438,7 +445,7 @@ method parseChunk(chunk: PNGHeader, png: PNG): bool =
|
|||||||
|
|
||||||
method parseChunk(chunk: PNGPalette, png: PNG): bool =
|
method parseChunk(chunk: PNGPalette, png: PNG): bool =
|
||||||
let paletteSize = chunk.length div 3
|
let paletteSize = chunk.length div 3
|
||||||
if paletteSize > 256: raise PNGError("palette size to big")
|
if paletteSize > 256: raise PNGFatal("palette size to big")
|
||||||
newSeq(chunk.palette, paletteSize)
|
newSeq(chunk.palette, paletteSize)
|
||||||
for px in mitems(chunk.palette):
|
for px in mitems(chunk.palette):
|
||||||
px.r = chr(chunk.readByte())
|
px.r = chr(chunk.readByte())
|
||||||
@ -447,7 +454,7 @@ method parseChunk(chunk: PNGPalette, png: PNG): bool =
|
|||||||
px.a = chr(255)
|
px.a = chr(255)
|
||||||
result = true
|
result = true
|
||||||
|
|
||||||
proc numChannels(colorType: PNGcolorType): int =
|
proc numChannels(colorType: PNGColorType): int =
|
||||||
case colorType
|
case colorType
|
||||||
of LCT_GREY: result = 1
|
of LCT_GREY: result = 1
|
||||||
of LCT_RGB : result = 3
|
of LCT_RGB : result = 3
|
||||||
@ -455,7 +462,7 @@ proc numChannels(colorType: PNGcolorType): int =
|
|||||||
of LCT_GREY_ALPHA: result = 2
|
of LCT_GREY_ALPHA: result = 2
|
||||||
of LCT_RGBA: result = 4
|
of LCT_RGBA: result = 4
|
||||||
|
|
||||||
proc LCTBPP(colorType: PNGcolorType, bitDepth: int): int =
|
proc LCTBPP(colorType: PNGColorType, bitDepth: int): int =
|
||||||
# bits per pixel is amount of channels * bits per channel
|
# bits per pixel is amount of channels * bits per channel
|
||||||
result = numChannels(colorType) * bitDepth
|
result = numChannels(colorType) * bitDepth
|
||||||
|
|
||||||
@ -473,7 +480,7 @@ proc idatRawSize(w, h: int, header: PNGHeader): int =
|
|||||||
proc getRawSize(w, h: int, color: PNGColorMode): int =
|
proc getRawSize(w, h: int, color: PNGColorMode): int =
|
||||||
result = (w * h * getBPP(color) + 7) div 8
|
result = (w * h * getBPP(color) + 7) div 8
|
||||||
|
|
||||||
#proc getRawSizeLct(w, h: int, colorType: PNGcolorType, bitDepth: int): int =
|
#proc getRawSizeLct(w, h: int, colorType: PNGColorType, bitDepth: int): int =
|
||||||
# result = (w * h * LCTBPP(colorType, bitDepth) + 7) div 8
|
# result = (w * h * LCTBPP(colorType, bitDepth) + 7) div 8
|
||||||
|
|
||||||
method validateChunk(chunk: PNGData, png: PNG): bool =
|
method validateChunk(chunk: PNGData, png: PNG): bool =
|
||||||
@ -495,7 +502,7 @@ method validateChunk(chunk: PNGData, png: PNG): bool =
|
|||||||
if w > 1: predict += idatRawSize((w + 0) div 2, (h + 1) div 2, header) + (h + 1) div 2
|
if w > 1: predict += idatRawSize((w + 0) div 2, (h + 1) div 2, header) + (h + 1) div 2
|
||||||
predict += idatRawSize((w + 0) div 1, (h + 0) div 2, header) + (h + 0) div 2
|
predict += idatRawSize((w + 0) div 1, (h + 0) div 2, header) + (h + 0) div 2
|
||||||
|
|
||||||
if chunk.idat.len != predict: raise PNGError("Decompress size doesn't match predict")
|
if chunk.idat.len != predict: raise PNGFatal("Decompress size doesn't match predict")
|
||||||
result = true
|
result = true
|
||||||
|
|
||||||
method parseChunk(chunk: PNGData, png: PNG): bool =
|
method parseChunk(chunk: PNGData, png: PNG): bool =
|
||||||
@ -513,23 +520,23 @@ method parseChunk(chunk: PNGTrans, png: PNG): bool =
|
|||||||
if plte == nil: return false
|
if plte == nil: return false
|
||||||
# error: more alpha values given than there are palette entries
|
# error: more alpha values given than there are palette entries
|
||||||
if chunk.length > plte.palette.len:
|
if chunk.length > plte.palette.len:
|
||||||
raise PNGError("more alpha value than palette entries")
|
raise PNGFatal("more alpha value than palette entries")
|
||||||
#can contain fewer values than palette entries
|
#can contain fewer values than palette entries
|
||||||
for i in 0..chunk.length-1: plte.palette[i].a = chr(chunk.readByte())
|
for i in 0..chunk.length-1: plte.palette[i].a = chr(chunk.readByte())
|
||||||
elif header.colorType == LCT_GREY:
|
elif header.colorType == LCT_GREY:
|
||||||
# error: this chunk must be 2 bytes for greyscale image
|
# error: this chunk must be 2 bytes for greyscale image
|
||||||
if chunk.length != 2: raise PNGError("tRNS must be 2 bytes")
|
if chunk.length != 2: raise PNGFatal("tRNS must be 2 bytes")
|
||||||
chunk.keyR = chunk.readInt16()
|
chunk.keyR = chunk.readInt16()
|
||||||
chunk.keyG = chunk.keyR
|
chunk.keyG = chunk.keyR
|
||||||
chunk.keyB = chunk.keyR
|
chunk.keyB = chunk.keyR
|
||||||
elif header.colorType == LCT_RGB:
|
elif header.colorType == LCT_RGB:
|
||||||
# error: this chunk must be 6 bytes for RGB image
|
# error: this chunk must be 6 bytes for RGB image
|
||||||
if chunk.length != 6: raise PNGError("tRNS must be 6 bytes")
|
if chunk.length != 6: raise PNGFatal("tRNS must be 6 bytes")
|
||||||
chunk.keyR = chunk.readInt16()
|
chunk.keyR = chunk.readInt16()
|
||||||
chunk.keyG = chunk.readInt16()
|
chunk.keyG = chunk.readInt16()
|
||||||
chunk.keyB = chunk.readInt16()
|
chunk.keyB = chunk.readInt16()
|
||||||
else:
|
else:
|
||||||
raise PNGError("tRNS chunk not allowed for other color models")
|
raise PNGFatal("tRNS chunk not allowed for other color models")
|
||||||
|
|
||||||
result = true
|
result = true
|
||||||
|
|
||||||
@ -537,19 +544,19 @@ method parseChunk(chunk: PNGBackground, png: PNG): bool =
|
|||||||
var header = PNGHeader(png.getChunk(IHDR))
|
var header = PNGHeader(png.getChunk(IHDR))
|
||||||
if header.colorType == LCT_PALETTE:
|
if header.colorType == LCT_PALETTE:
|
||||||
# error: this chunk must be 1 byte for indexed color image
|
# error: this chunk must be 1 byte for indexed color image
|
||||||
if chunk.length != 1: raise PNGError("bkgd must be 1 byte")
|
if chunk.length != 1: raise PNGFatal("bkgd must be 1 byte")
|
||||||
chunk.bkgdR = chunk.readByte()
|
chunk.bkgdR = chunk.readByte()
|
||||||
chunk.bkgdG = chunk.bkgdR
|
chunk.bkgdG = chunk.bkgdR
|
||||||
chunk.bkgdB = chunk.bkgdR
|
chunk.bkgdB = chunk.bkgdR
|
||||||
elif header.colorType in {LCT_GREY, LCT_GREY_ALPHA}:
|
elif header.colorType in {LCT_GREY, LCT_GREY_ALPHA}:
|
||||||
# error: this chunk must be 2 bytes for greyscale image
|
# error: this chunk must be 2 bytes for greyscale image
|
||||||
if chunk.length != 2: raise PNGError("bkgd must be 2 byte")
|
if chunk.length != 2: raise PNGFatal("bkgd must be 2 byte")
|
||||||
chunk.bkgdR = chunk.readInt16()
|
chunk.bkgdR = chunk.readInt16()
|
||||||
chunk.bkgdG = chunk.bkgdR
|
chunk.bkgdG = chunk.bkgdR
|
||||||
chunk.bkgdB = chunk.bkgdR
|
chunk.bkgdB = chunk.bkgdR
|
||||||
elif header.colorType in {LCT_RGB, LCT_RGBA}:
|
elif header.colorType in {LCT_RGB, LCT_RGBA}:
|
||||||
# error: this chunk must be 6 bytes for greyscale image
|
# error: this chunk must be 6 bytes for greyscale image
|
||||||
if chunk.length != 6: raise PNGError("bkgd must be 6 byte")
|
if chunk.length != 6: raise PNGFatal("bkgd must be 6 byte")
|
||||||
chunk.bkgdR = chunk.readInt16()
|
chunk.bkgdR = chunk.readInt16()
|
||||||
chunk.bkgdG = chunk.readInt16()
|
chunk.bkgdG = chunk.readInt16()
|
||||||
chunk.bkgdB = chunk.readInt16()
|
chunk.bkgdB = chunk.readInt16()
|
||||||
@ -563,17 +570,17 @@ proc initChunk(chunk: PNGChunk, chunkType: PNGChunkType, data: string, crc: uint
|
|||||||
chunk.pos = 0
|
chunk.pos = 0
|
||||||
|
|
||||||
method validateChunk(chunk: PNGTime, png: PNG): bool =
|
method validateChunk(chunk: PNGTime, png: PNG): bool =
|
||||||
if chunk.year < 0 or chunk.year > 65535: raise PNGError("invalid year range[0..65535]")
|
if chunk.year < 0 or chunk.year > 65535: raise PNGFatal("invalid year range[0..65535]")
|
||||||
if chunk.month < 1 or chunk.month > 12: raise PNGError("invalid month range[1..12]")
|
if chunk.month < 1 or chunk.month > 12: raise PNGFatal("invalid month range[1..12]")
|
||||||
if chunk.day < 1 or chunk.day > 31: raise PNGError("invalid day range[1..32]")
|
if chunk.day < 1 or chunk.day > 31: raise PNGFatal("invalid day range[1..32]")
|
||||||
if chunk.hour < 0 or chunk.hour > 23: raise PNGError("invalid hour range[0..23]")
|
if chunk.hour < 0 or chunk.hour > 23: raise PNGFatal("invalid hour range[0..23]")
|
||||||
if chunk.minute < 0 or chunk.minute > 59: raise PNGError("invalid minute range[0..59]")
|
if chunk.minute < 0 or chunk.minute > 59: raise PNGFatal("invalid minute range[0..59]")
|
||||||
#to allow for leap seconds
|
#to allow for leap seconds
|
||||||
if chunk.second < 0 or chunk.second > 60: raise PNGError("invalid second range[0..60]")
|
if chunk.second < 0 or chunk.second > 60: raise PNGFatal("invalid second range[0..60]")
|
||||||
result = true
|
result = true
|
||||||
|
|
||||||
method parseChunk(chunk: PNGTime, png: PNG): bool =
|
method parseChunk(chunk: PNGTime, png: PNG): bool =
|
||||||
if chunk.length != 7: raise PNGError("tIME must be 7 bytes")
|
if chunk.length != 7: raise PNGFatal("tIME must be 7 bytes")
|
||||||
chunk.year = chunk.readInt16()
|
chunk.year = chunk.readInt16()
|
||||||
chunk.month = chunk.readByte()
|
chunk.month = chunk.readByte()
|
||||||
chunk.day = chunk.readByte()
|
chunk.day = chunk.readByte()
|
||||||
@ -583,7 +590,7 @@ method parseChunk(chunk: PNGTime, png: PNG): bool =
|
|||||||
result = true
|
result = true
|
||||||
|
|
||||||
method parseChunk(chunk: PNGPhys, png: PNG): bool =
|
method parseChunk(chunk: PNGPhys, png: PNG): bool =
|
||||||
if chunk.length != 9: raise PNGError("pHYs must be 9 bytes")
|
if chunk.length != 9: raise PNGFatal("pHYs must be 9 bytes")
|
||||||
chunk.physX = chunk.readInt32()
|
chunk.physX = chunk.readInt32()
|
||||||
chunk.physY = chunk.readInt32()
|
chunk.physY = chunk.readInt32()
|
||||||
chunk.unit = chunk.readByte()
|
chunk.unit = chunk.readByte()
|
||||||
@ -591,13 +598,13 @@ method parseChunk(chunk: PNGPhys, png: PNG): bool =
|
|||||||
|
|
||||||
method validateChunk(chunk: PNGText, png: PNG): bool =
|
method validateChunk(chunk: PNGText, png: PNG): bool =
|
||||||
if(chunk.keyword.len < 1) or (chunk.keyword.len > 79):
|
if(chunk.keyword.len < 1) or (chunk.keyword.len > 79):
|
||||||
raise PNGError("keyword too short or too long")
|
raise PNGFatal("keyword too short or too long")
|
||||||
result = true
|
result = true
|
||||||
|
|
||||||
method parseChunk(chunk: PNGText, png: PNG): bool =
|
method parseChunk(chunk: PNGText, png: PNG): bool =
|
||||||
var len = 0
|
var len = 0
|
||||||
while(len < chunk.length) and (chunk.data[len] != chr(0)): inc len
|
while(len < chunk.length) and (chunk.data[len] != chr(0)): inc len
|
||||||
if(len < 1) or (len > 79): raise PNGError("keyword too short or too long")
|
if(len < 1) or (len > 79): raise PNGFatal("keyword too short or too long")
|
||||||
chunk.keyword = chunk.data.substr(0, len)
|
chunk.keyword = chunk.data.substr(0, len)
|
||||||
|
|
||||||
var textBegin = len + 1 # skip keyword null terminator
|
var textBegin = len + 1 # skip keyword null terminator
|
||||||
@ -606,17 +613,17 @@ method parseChunk(chunk: PNGText, png: PNG): bool =
|
|||||||
|
|
||||||
method validateChunk(chunk: PNGZtxt, png: PNG): bool =
|
method validateChunk(chunk: PNGZtxt, png: PNG): bool =
|
||||||
if(chunk.keyword.len < 1) or (chunk.keyword.len > 79):
|
if(chunk.keyword.len < 1) or (chunk.keyword.len > 79):
|
||||||
raise PNGError("keyword too short or too long")
|
raise PNGFatal("keyword too short or too long")
|
||||||
result = true
|
result = true
|
||||||
|
|
||||||
method parseChunk(chunk: PNGZtxt, png: PNG): bool =
|
method parseChunk(chunk: PNGZtxt, png: PNG): bool =
|
||||||
var len = 0
|
var len = 0
|
||||||
while(len < chunk.length) and (chunk.data[len] != chr(0)): inc len
|
while(len < chunk.length) and (chunk.data[len] != chr(0)): inc len
|
||||||
if(len < 1) or (len > 79): raise PNGError("keyword too short or too long")
|
if(len < 1) or (len > 79): raise PNGFatal("keyword too short or too long")
|
||||||
chunk.keyword = chunk.data.substr(0, len)
|
chunk.keyword = chunk.data.substr(0, len)
|
||||||
|
|
||||||
var compMethod = ord(chunk.data[len + 1]) # skip keyword null terminator
|
var compMethod = ord(chunk.data[len + 1]) # skip keyword null terminator
|
||||||
if compMethod != 0: raise PNGError("unsupported comp method")
|
if compMethod != 0: raise PNGFatal("unsupported comp method")
|
||||||
|
|
||||||
var nz = nzInflateInit(chunk.data.substr(len + 2))
|
var nz = nzInflateInit(chunk.data.substr(len + 2))
|
||||||
nz.ignoreAdler32 = PNGDecoder(png.settings).ignoreAdler32
|
nz.ignoreAdler32 = PNGDecoder(png.settings).ignoreAdler32
|
||||||
@ -626,22 +633,22 @@ method parseChunk(chunk: PNGZtxt, png: PNG): bool =
|
|||||||
|
|
||||||
method validateChunk(chunk: PNGItxt, png: PNG): bool =
|
method validateChunk(chunk: PNGItxt, png: PNG): bool =
|
||||||
if(chunk.keyword.len < 1) or (chunk.keyword.len > 79):
|
if(chunk.keyword.len < 1) or (chunk.keyword.len > 79):
|
||||||
raise PNGError("keyword too short or too long")
|
raise PNGFatal("keyword too short or too long")
|
||||||
result = true
|
result = true
|
||||||
|
|
||||||
method parseChunk(chunk: PNGItxt, png: PNG): bool =
|
method parseChunk(chunk: PNGItxt, png: PNG): bool =
|
||||||
if chunk.length < 5: raise PNGError("iTXt len too short")
|
if chunk.length < 5: raise PNGFatal("iTXt len too short")
|
||||||
|
|
||||||
var len = 0
|
var len = 0
|
||||||
while(len < chunk.length) and (chunk.data[len] != chr(0)): inc len
|
while(len < chunk.length) and (chunk.data[len] != chr(0)): inc len
|
||||||
|
|
||||||
if(len + 3) >= chunk.length: raise PNGError("no null termination char, corrupt?")
|
if(len + 3) >= chunk.length: raise PNGFatal("no null termination char, corrupt?")
|
||||||
if(len < 1) or (len > 79): raise PNGError("keyword too short or too long")
|
if(len < 1) or (len > 79): raise PNGFatal("keyword too short or too long")
|
||||||
chunk.keyword = chunk.data.substr(0, len)
|
chunk.keyword = chunk.data.substr(0, len)
|
||||||
|
|
||||||
var compressed = ord(chunk.data[len + 1]) == 1 # skip keyword null terminator
|
var compressed = ord(chunk.data[len + 1]) == 1 # skip keyword null terminator
|
||||||
var compMethod = ord(chunk.data[len + 2])
|
var compMethod = ord(chunk.data[len + 2])
|
||||||
if compMethod != 0: raise PNGError("unsupported comp method")
|
if compMethod != 0: raise PNGFatal("unsupported comp method")
|
||||||
|
|
||||||
len = 0
|
len = 0
|
||||||
var i = len + 3
|
var i = len + 3
|
||||||
@ -669,12 +676,12 @@ method parseChunk(chunk: PNGItxt, png: PNG): bool =
|
|||||||
result = true
|
result = true
|
||||||
|
|
||||||
method parseChunk(chunk: PNGGamma, png: PNG): bool =
|
method parseChunk(chunk: PNGGamma, png: PNG): bool =
|
||||||
if chunk.length != 4: raise PNGError("invalid gAMA length")
|
if chunk.length != 4: raise PNGFatal("invalid gAMA length")
|
||||||
chunk.gamma = chunk.readInt32()
|
chunk.gamma = chunk.readInt32()
|
||||||
result = true
|
result = true
|
||||||
|
|
||||||
method parseChunk(chunk: PNGChroma, png: PNG): bool =
|
method parseChunk(chunk: PNGChroma, png: PNG): bool =
|
||||||
if chunk.length != 32: raise PNGError("invalid Chroma length")
|
if chunk.length != 32: raise PNGFatal("invalid Chroma length")
|
||||||
chunk.whitePointX = chunk.readInt32()
|
chunk.whitePointX = chunk.readInt32()
|
||||||
chunk.whitePointY = chunk.readInt32()
|
chunk.whitePointY = chunk.readInt32()
|
||||||
chunk.redX = chunk.readInt32()
|
chunk.redX = chunk.readInt32()
|
||||||
@ -686,23 +693,23 @@ method parseChunk(chunk: PNGChroma, png: PNG): bool =
|
|||||||
result = true
|
result = true
|
||||||
|
|
||||||
method parseChunk(chunk: PNGStandarRGB, png: PNG): bool =
|
method parseChunk(chunk: PNGStandarRGB, png: PNG): bool =
|
||||||
if chunk.length != 1: raise PNGError("invalid sRGB length")
|
if chunk.length != 1: raise PNGFatal("invalid sRGB length")
|
||||||
chunk.renderingIntent = chunk.readByte()
|
chunk.renderingIntent = chunk.readByte()
|
||||||
result = true
|
result = true
|
||||||
|
|
||||||
method validateChunk(chunk: PNGICCProfile, png: PNG): bool =
|
method validateChunk(chunk: PNGICCProfile, png: PNG): bool =
|
||||||
if(chunk.profileName.len < 1) or (chunk.profileName.len > 79):
|
if(chunk.profileName.len < 1) or (chunk.profileName.len > 79):
|
||||||
raise PNGError("keyword too short or too long")
|
raise PNGFatal("keyword too short or too long")
|
||||||
result = true
|
result = true
|
||||||
|
|
||||||
method parseChunk(chunk: PNGICCProfile, png: PNG): bool =
|
method parseChunk(chunk: PNGICCProfile, png: PNG): bool =
|
||||||
var len = 0
|
var len = 0
|
||||||
while(len < chunk.length) and (chunk.data[len] != chr(0)): inc len
|
while(len < chunk.length) and (chunk.data[len] != chr(0)): inc len
|
||||||
if(len < 1) or (len > 79): raise PNGError("keyword too short or too long")
|
if(len < 1) or (len > 79): raise PNGFatal("keyword too short or too long")
|
||||||
chunk.profileName = chunk.data.substr(0, len)
|
chunk.profileName = chunk.data.substr(0, len)
|
||||||
|
|
||||||
var compMethod = ord(chunk.data[len + 1]) # skip keyword null terminator
|
var compMethod = ord(chunk.data[len + 1]) # skip keyword null terminator
|
||||||
if compMethod != 0: raise PNGError("unsupported comp method")
|
if compMethod != 0: raise PNGFatal("unsupported comp method")
|
||||||
|
|
||||||
var nz = nzInflateInit(chunk.data.substr(len + 2))
|
var nz = nzInflateInit(chunk.data.substr(len + 2))
|
||||||
nz.ignoreAdler32 = PNGDecoder(png.settings).ignoreAdler32
|
nz.ignoreAdler32 = PNGDecoder(png.settings).ignoreAdler32
|
||||||
@ -712,15 +719,15 @@ method parseChunk(chunk: PNGICCProfile, png: PNG): bool =
|
|||||||
method parseChunk(chunk: PNGSPalette, png: PNG): bool =
|
method parseChunk(chunk: PNGSPalette, png: PNG): bool =
|
||||||
var len = 0
|
var len = 0
|
||||||
while(len < chunk.length) and (chunk.data[len] != chr(0)): inc len
|
while(len < chunk.length) and (chunk.data[len] != chr(0)): inc len
|
||||||
if(len < 1) or (len > 79): raise PNGError("keyword too short or too long")
|
if(len < 1) or (len > 79): raise PNGFatal("keyword too short or too long")
|
||||||
chunk.paletteName = chunk.data.substr(0, len)
|
chunk.paletteName = chunk.data.substr(0, len)
|
||||||
chunk.setPosition(len + 1)
|
chunk.setPosition(len + 1)
|
||||||
chunk.sampleDepth = chunk.readByte()
|
chunk.sampleDepth = chunk.readByte()
|
||||||
if chunk.sampleDepth notin {8, 16}: raise PNGError("palette sample depth error")
|
if chunk.sampleDepth notin {8, 16}: raise PNGFatal("palette sample depth error")
|
||||||
|
|
||||||
let remainingLength = (chunk.length - (len + 2))
|
let remainingLength = (chunk.length - (len + 2))
|
||||||
if chunk.sampleDepth == 8:
|
if chunk.sampleDepth == 8:
|
||||||
if (remainingLength mod 6) != 0: raise PNGError("palette length not divisible by 6")
|
if (remainingLength mod 6) != 0: raise PNGFatal("palette length not divisible by 6")
|
||||||
let numSamples = remainingLength div 6
|
let numSamples = remainingLength div 6
|
||||||
newSeq(chunk.palette, numSamples)
|
newSeq(chunk.palette, numSamples)
|
||||||
for p in mitems(chunk.palette):
|
for p in mitems(chunk.palette):
|
||||||
@ -730,7 +737,7 @@ method parseChunk(chunk: PNGSPalette, png: PNG): bool =
|
|||||||
p.alpha = chunk.readByte()
|
p.alpha = chunk.readByte()
|
||||||
p.frequency = chunk.readInt16()
|
p.frequency = chunk.readInt16()
|
||||||
else: # chunk.sampleDepth == 16:
|
else: # chunk.sampleDepth == 16:
|
||||||
if (remainingLength mod 10) != 0: raise PNGError("palette length not divisible by 10")
|
if (remainingLength mod 10) != 0: raise PNGFatal("palette length not divisible by 10")
|
||||||
let numSamples = remainingLength div 10
|
let numSamples = remainingLength div 10
|
||||||
newSeq(chunk.palette, numSamples)
|
newSeq(chunk.palette, numSamples)
|
||||||
for p in mitems(chunk.palette):
|
for p in mitems(chunk.palette):
|
||||||
@ -743,9 +750,9 @@ method parseChunk(chunk: PNGSPalette, png: PNG): bool =
|
|||||||
result = true
|
result = true
|
||||||
|
|
||||||
method parseChunk(chunk: PNGHist, png: PNG): bool =
|
method parseChunk(chunk: PNGHist, png: PNG): bool =
|
||||||
if not png.hasChunk(PLTE): raise PNGError("Histogram need PLTE")
|
if not png.hasChunk(PLTE): raise PNGFatal("Histogram need PLTE")
|
||||||
var plte = PNGPalette(png.getChunk(PLTE))
|
var plte = PNGPalette(png.getChunk(PLTE))
|
||||||
if plte.palette.len != (chunk.length div 2): raise PNGError("invalid histogram length")
|
if plte.palette.len != (chunk.length div 2): raise PNGFatal("invalid histogram length")
|
||||||
newSeq(chunk.histogram, plte.palette.len)
|
newSeq(chunk.histogram, plte.palette.len)
|
||||||
for i in 0..chunk.histogram.high:
|
for i in 0..chunk.histogram.high:
|
||||||
chunk.histogram[i] = chunk.readInt16()
|
chunk.histogram[i] = chunk.readInt16()
|
||||||
@ -761,11 +768,11 @@ method parseChunk(chunk: PNGSbit, png: PNG): bool =
|
|||||||
of LCT_PALETTE: expectedLen = 3
|
of LCT_PALETTE: expectedLen = 3
|
||||||
of LCT_GREY_ALPHA: expectedLen = 2
|
of LCT_GREY_ALPHA: expectedLen = 2
|
||||||
of LCT_RGBA: expectedLen = 4
|
of LCT_RGBA: expectedLen = 4
|
||||||
if chunk.length != expectedLen: raise PNGError("invalid sBIT length")
|
if chunk.length != expectedLen: raise PNGFatal("invalid sBIT length")
|
||||||
var expectedDepth = 8 #LCT_PALETTE
|
var expectedDepth = 8 #LCT_PALETTE
|
||||||
if header.colorType != LCT_PALETTE: expectedDepth = header.bitDepth
|
if header.colorType != LCT_PALETTE: expectedDepth = header.bitDepth
|
||||||
for c in chunk.data:
|
for c in chunk.data:
|
||||||
if (ord(c) == 0) or (ord(c) > expectedDepth): raise PNGError("invalid sBIT value")
|
if (ord(c) == 0) or (ord(c) > expectedDepth): raise PNGFatal("invalid sBIT value")
|
||||||
|
|
||||||
result = true
|
result = true
|
||||||
|
|
||||||
@ -873,38 +880,38 @@ proc parsePNG(s: Stream, settings: PNGDecoder): PNG =
|
|||||||
|
|
||||||
let signature = s.readStr(8)
|
let signature = s.readStr(8)
|
||||||
if signature != PNGSignature:
|
if signature != PNGSignature:
|
||||||
raise PNGError("signature mismatch")
|
raise PNGFatal("signature mismatch")
|
||||||
|
|
||||||
while not s.atEnd():
|
while not s.atEnd():
|
||||||
let length = s.readInt32BE()
|
let length = s.readInt32BE()
|
||||||
let chunkType = PNGChunkType(s.readInt32BE())
|
let chunkType = PNGChunkType(s.readInt32BE())
|
||||||
|
|
||||||
let data = if length == 0: "" else: s.readStr(length)
|
let data = if length <= 0: "" else: s.readStr(length)
|
||||||
let crc = cast[uint32](s.readInt32BE())
|
let crc = cast[uint32](s.readInt32BE())
|
||||||
let calculatedCRC = crc32(crc32(0, $chunkType), data)
|
let calculatedCRC = crc32(crc32(0, $chunkType), data)
|
||||||
if calculatedCRC != crc and not PNGDecoder(png.settings).ignoreCRC:
|
if calculatedCRC != crc and not PNGDecoder(png.settings).ignoreCRC:
|
||||||
raise PNGError("wrong crc for: " & $chunkType)
|
raise PNGFatal("wrong crc for: " & $chunkType)
|
||||||
var chunk = png.createChunk(chunkType, data, crc)
|
var chunk = png.createChunk(chunkType, data, crc)
|
||||||
|
|
||||||
if chunkType != IDAT and chunk != nil:
|
if chunkType != IDAT and chunk != nil:
|
||||||
if not chunk.parseChunk(png): raise PNGError("error parse chunk: " & $chunkType)
|
if not chunk.parseChunk(png): raise PNGFatal("error parse chunk: " & $chunkType)
|
||||||
if not chunk.validateChunk(png): raise PNGError("invalid chunk: " & $chunkType)
|
if not chunk.validateChunk(png): raise PNGFatal("invalid chunk: " & $chunkType)
|
||||||
if chunk != nil:
|
if chunk != nil:
|
||||||
if chunkType == fcTL or chunkType == fdAT:
|
if chunkType == fcTL or chunkType == fdAT:
|
||||||
png.apngChunks.add APNGFrameChunk(chunk)
|
png.apngChunks.add APNGFrameChunk(chunk)
|
||||||
else: png.chunks.add chunk
|
else: png.chunks.add chunk
|
||||||
if chunkType == IEND: break
|
if chunkType == IEND: break
|
||||||
|
|
||||||
if not png.hasChunk(IHDR): raise PNGError("no IHDR found")
|
if not png.hasChunk(IHDR): raise PNGFatal("no IHDR found")
|
||||||
if not png.hasChunk(IDAT): raise PNGError("no IDAT found")
|
if not png.hasChunk(IDAT): raise PNGFatal("no IDAT found")
|
||||||
var header = PNGHeader(png.getChunk(IHDR))
|
var header = PNGHeader(png.getChunk(IHDR))
|
||||||
if header.colorType == LCT_PALETTE and not png.hasChunk(PLTE):
|
if header.colorType == LCT_PALETTE and not png.hasChunk(PLTE):
|
||||||
raise PNGError("expected PLTE not found")
|
raise PNGFatal("expected PLTE not found")
|
||||||
|
|
||||||
# IDAT get special treatment because it can appear in multiple chunk
|
# IDAT get special treatment because it can appear in multiple chunk
|
||||||
var idat = PNGData(png.getChunk(IDAT))
|
var idat = PNGData(png.getChunk(IDAT))
|
||||||
if not idat.parseChunk(png): raise PNGError("IDAT parse error")
|
if not idat.parseChunk(png): raise PNGFatal("IDAT parse error")
|
||||||
if not idat.validateChunk(png): raise PNGError("bad IDAT")
|
if not idat.validateChunk(png): raise PNGFatal("bad IDAT")
|
||||||
result = png
|
result = png
|
||||||
|
|
||||||
proc postProcessScanLines[T](png: PNG; header: PNGHeader, w, h: int; input, output: var openArray[T]) =
|
proc postProcessScanLines[T](png: PNG; header: PNGHeader, w, h: int; input, output: var openArray[T]) =
|
||||||
@ -1526,14 +1533,14 @@ proc getColorRGBA16[T](mode: PNGColorMode): convertRGBA16[T] =
|
|||||||
elif mode.colorType == LCT_RGB: return RGBA16FromRGB[T]
|
elif mode.colorType == LCT_RGB: return RGBA16FromRGB[T]
|
||||||
elif mode.colorType == LCT_GREY_ALPHA: return RGBA16FromGreyAlpha[T]
|
elif mode.colorType == LCT_GREY_ALPHA: return RGBA16FromGreyAlpha[T]
|
||||||
elif mode.colorType == LCT_RGBA: return RGBA16FromRGBA[T]
|
elif mode.colorType == LCT_RGBA: return RGBA16FromRGBA[T]
|
||||||
else: raise PNGError("unsupported converter16")
|
else: raise PNGFatal("unsupported converter16")
|
||||||
|
|
||||||
proc getPixelRGBA16[T](mode: PNGColorMode): pixelRGBA16[T] =
|
proc getPixelRGBA16[T](mode: PNGColorMode): pixelRGBA16[T] =
|
||||||
if mode.colorType == LCT_GREY: return RGBA16ToGrey[T]
|
if mode.colorType == LCT_GREY: return RGBA16ToGrey[T]
|
||||||
elif mode.colorType == LCT_RGB: return RGBA16ToRGB[T]
|
elif mode.colorType == LCT_RGB: return RGBA16ToRGB[T]
|
||||||
elif mode.colorType == LCT_GREY_ALPHA: return RGBA16ToGreyAlpha[T]
|
elif mode.colorType == LCT_GREY_ALPHA: return RGBA16ToGreyAlpha[T]
|
||||||
elif mode.colorType == LCT_RGBA: return RGBA16ToRGBA[T]
|
elif mode.colorType == LCT_RGBA: return RGBA16ToRGBA[T]
|
||||||
else: raise PNGError("unsupported pixel16 converter")
|
else: raise PNGFatal("unsupported pixel16 converter")
|
||||||
|
|
||||||
proc getColorRGBA8[T](mode: PNGColorMode): convertRGBA8[T] =
|
proc getColorRGBA8[T](mode: PNGColorMode): convertRGBA8[T] =
|
||||||
if mode.colorType == LCT_GREY:
|
if mode.colorType == LCT_GREY:
|
||||||
@ -1552,7 +1559,7 @@ proc getColorRGBA8[T](mode: PNGColorMode): convertRGBA8[T] =
|
|||||||
elif mode.colorType == LCT_RGBA:
|
elif mode.colorType == LCT_RGBA:
|
||||||
if mode.bitDepth == 8: return RGBA8FromRGBA8[T]
|
if mode.bitDepth == 8: return RGBA8FromRGBA8[T]
|
||||||
else: return RGBA8FromRGBA16[T]
|
else: return RGBA8FromRGBA16[T]
|
||||||
else: raise PNGError("unsupported converter8")
|
else: raise PNGFatal("unsupported converter8")
|
||||||
|
|
||||||
proc getPixelRGBA8[T](mode: PNGColorMode): pixelRGBA8[T] =
|
proc getPixelRGBA8[T](mode: PNGColorMode): pixelRGBA8[T] =
|
||||||
if mode.colorType == LCT_GREY:
|
if mode.colorType == LCT_GREY:
|
||||||
@ -1571,7 +1578,7 @@ proc getPixelRGBA8[T](mode: PNGColorMode): pixelRGBA8[T] =
|
|||||||
elif mode.colorType == LCT_RGBA:
|
elif mode.colorType == LCT_RGBA:
|
||||||
if mode.bitDepth == 8: return RGBA8ToRGBA8[T]
|
if mode.bitDepth == 8: return RGBA8ToRGBA8[T]
|
||||||
else: return RGBA8ToRGBA16[T]
|
else: return RGBA8ToRGBA16[T]
|
||||||
else: raise PNGError("unsupported pixel8 converter")
|
else: raise PNGFatal("unsupported pixel8 converter")
|
||||||
|
|
||||||
proc getConverterRGB[T](mode: PNGColorMode): convertRGBA[T] =
|
proc getConverterRGB[T](mode: PNGColorMode): convertRGBA[T] =
|
||||||
if mode.colorType == LCT_GREY:
|
if mode.colorType == LCT_GREY:
|
||||||
@ -1590,7 +1597,7 @@ proc getConverterRGB[T](mode: PNGColorMode): convertRGBA[T] =
|
|||||||
elif mode.colorType == LCT_RGBA:
|
elif mode.colorType == LCT_RGBA:
|
||||||
if mode.bitDepth == 8: return RGBFromRGBA8[T]
|
if mode.bitDepth == 8: return RGBFromRGBA8[T]
|
||||||
else: return RGBFromRGBA16[T]
|
else: return RGBFromRGBA16[T]
|
||||||
else: raise PNGError("unsupported RGB converter")
|
else: raise PNGFatal("unsupported RGB converter")
|
||||||
|
|
||||||
proc getConverterRGBA[T](mode: PNGColorMode): convertRGBA[T] =
|
proc getConverterRGBA[T](mode: PNGColorMode): convertRGBA[T] =
|
||||||
if mode.colorType == LCT_GREY:
|
if mode.colorType == LCT_GREY:
|
||||||
@ -1609,7 +1616,7 @@ proc getConverterRGBA[T](mode: PNGColorMode): convertRGBA[T] =
|
|||||||
elif mode.colorType == LCT_RGBA:
|
elif mode.colorType == LCT_RGBA:
|
||||||
if mode.bitDepth == 8: return RGBAFromRGBA8[T]
|
if mode.bitDepth == 8: return RGBAFromRGBA8[T]
|
||||||
else: return RGBAFromRGBA16[T]
|
else: return RGBAFromRGBA16[T]
|
||||||
else: raise PNGError("unsupported RGBA converter")
|
else: raise PNGFatal("unsupported RGBA converter")
|
||||||
|
|
||||||
proc convert*[T](output: var openArray[T], input: openArray[T], modeOut, modeIn: PNGColorMode, numPixels: int) =
|
proc convert*[T](output: var openArray[T], input: openArray[T], modeOut, modeIn: PNGColorMode, numPixels: int) =
|
||||||
var tree: ColorTree8
|
var tree: ColorTree8
|
||||||
@ -1653,11 +1660,11 @@ proc convert*[T](output: var openArray[T], input: openArray[T], modeOut, modeIn:
|
|||||||
cvt(p, input, px, modeIn)
|
cvt(p, input, px, modeIn)
|
||||||
pxl(p, output, px, modeOut, tree)
|
pxl(p, output, px, modeOut, tree)
|
||||||
|
|
||||||
proc convert*(png: PNG, colorType: PNGcolorType, bitDepth: int): PNGResult =
|
proc convert*(png: PNG, colorType: PNGColorType, bitDepth: int): PNGResult =
|
||||||
# TODO: check if this works according to the statement in the documentation: "The converter can convert
|
# TODO: check if this works according to the statement in the documentation: "The converter can convert
|
||||||
# from greyscale input color type, to 8-bit greyscale or greyscale with alpha"
|
# from greyscale input color type, to 8-bit greyscale or greyscale with alpha"
|
||||||
# if(colorType notin {LCT_RGB, LCT_RGBA}) and (bitDepth != 8):
|
# if(colorType notin {LCT_RGB, LCT_RGBA}) and (bitDepth != 8):
|
||||||
# raise PNGError("unsupported color mode conversion")
|
# raise PNGFatal("unsupported color mode conversion")
|
||||||
|
|
||||||
let header = PNGHeader(png.getChunk(IHDR))
|
let header = PNGHeader(png.getChunk(IHDR))
|
||||||
let modeIn = png.getColorMode()
|
let modeIn = png.getColorMode()
|
||||||
@ -1678,7 +1685,7 @@ proc convert*(png: PNG, colorType: PNGcolorType, bitDepth: int): PNGResult =
|
|||||||
png.pixels.toOpenArray(0, png.pixels.len-1),
|
png.pixels.toOpenArray(0, png.pixels.len-1),
|
||||||
modeOut, modeIn, numPixels)
|
modeOut, modeIn, numPixels)
|
||||||
|
|
||||||
proc convert*(png: PNG, colorType: PNGcolorType, bitDepth: int, ctl: APNGFrameControl, data: string): APNGFrame =
|
proc convert*(png: PNG, colorType: PNGColorType, bitDepth: int, ctl: APNGFrameControl, data: string): APNGFrame =
|
||||||
let modeIn = png.getColorMode()
|
let modeIn = png.getColorMode()
|
||||||
let modeOut = newColorMode(colorType, bitDepth)
|
let modeOut = newColorMode(colorType, bitDepth)
|
||||||
let size = getRawSize(ctl.width, ctl.height, modeOut)
|
let size = getRawSize(ctl.width, ctl.height, modeOut)
|
||||||
@ -1707,7 +1714,7 @@ type
|
|||||||
png: PNG
|
png: PNG
|
||||||
result: PNGResult
|
result: PNGResult
|
||||||
|
|
||||||
proc processingAPNG(apng: APNG, colorType: PNGcolorType, bitDepth: int) =
|
proc processingAPNG(apng: APNG, colorType: PNGColorType, bitDepth: int) =
|
||||||
let header = PNGHeader(apng.png.getChunk(IHDR))
|
let header = PNGHeader(apng.png.getChunk(IHDR))
|
||||||
var
|
var
|
||||||
actl = APNGAnimationControl(apng.png.getChunk(acTL))
|
actl = APNGAnimationControl(apng.png.getChunk(acTL))
|
||||||
@ -1736,7 +1743,7 @@ proc processingAPNG(apng: APNG, colorType: PNGcolorType, bitDepth: int) =
|
|||||||
lastChunkType = fdAT
|
lastChunkType = fdAT
|
||||||
|
|
||||||
if actl.numFrames == 0 or actl.numFrames != numFrames or actl.numFrames != frameData.len:
|
if actl.numFrames == 0 or actl.numFrames != numFrames or actl.numFrames != frameData.len:
|
||||||
raise PNGError("animation numFrames error")
|
raise PNGFatal("animation numFrames error")
|
||||||
|
|
||||||
apng.png.apngPixels = newSeqOfCap[string](numFrames)
|
apng.png.apngPixels = newSeqOfCap[string](numFrames)
|
||||||
|
|
||||||
@ -1746,9 +1753,9 @@ proc processingAPNG(apng: APNG, colorType: PNGcolorType, bitDepth: int) =
|
|||||||
if apng.png.firstFrameIsDefaultImage:
|
if apng.png.firstFrameIsDefaultImage:
|
||||||
let ctl = frameControl[0]
|
let ctl = frameControl[0]
|
||||||
if ctl.width != header.width or ctl.height != header.height:
|
if ctl.width != header.width or ctl.height != header.height:
|
||||||
raise PNGError("animation control error: dimension")
|
raise PNGFatal("animation control error: dimension")
|
||||||
if ctl.xOffset != 0 or ctl.xOffset != 0:
|
if ctl.xOffset != 0 or ctl.xOffset != 0:
|
||||||
raise PNGError("animation control error: offset")
|
raise PNGFatal("animation control error: offset")
|
||||||
|
|
||||||
if apng.result != nil:
|
if apng.result != nil:
|
||||||
var frame = new(APNGFrame)
|
var frame = new(APNGFrame)
|
||||||
@ -1772,9 +1779,9 @@ proc processingAPNG(apng: APNG, colorType: PNGcolorType, bitDepth: int) =
|
|||||||
frame.data = apng.png.apngPixels[^1]
|
frame.data = apng.png.apngPixels[^1]
|
||||||
apng.result.frames.add frame
|
apng.result.frames.add frame
|
||||||
|
|
||||||
proc decodePNG*(s: Stream, colorType: PNGcolorType, bitDepth: int, settings = PNGDecoder(nil)): PNGResult =
|
proc decodePNG*(s: Stream, colorType: PNGColorType, bitDepth: int, settings = PNGDecoder(nil)): PNGResult =
|
||||||
if not bitDepthAllowed(colorType, bitDepth):
|
if not bitDepthAllowed(colorType, bitDepth):
|
||||||
raise PNGError("colorType and bitDepth combination not allowed")
|
raise PNGFatal("colorType and bitDepth combination not allowed")
|
||||||
|
|
||||||
var png = s.parsePNG(settings)
|
var png = s.parsePNG(settings)
|
||||||
png.postProcessScanLines()
|
png.postProcessScanLines()
|
||||||
@ -1809,7 +1816,7 @@ when not defined(js):
|
|||||||
if s == nil: return nil
|
if s == nil: return nil
|
||||||
result = s.decodePNG(colorType, bitDepth, settings)
|
result = s.decodePNG(colorType, bitDepth, settings)
|
||||||
s.close()
|
s.close()
|
||||||
except:
|
except PNGError, IOError, NZError:
|
||||||
debugEcho getCurrentExceptionMsg()
|
debugEcho getCurrentExceptionMsg()
|
||||||
result = nil
|
result = nil
|
||||||
|
|
||||||
@ -1824,7 +1831,7 @@ proc decodePNG32*(input: string, settings = PNGDecoder(nil)): PNGResult =
|
|||||||
var s = newStringStream(input)
|
var s = newStringStream(input)
|
||||||
if s == nil: return nil
|
if s == nil: return nil
|
||||||
result = s.decodePNG(LCT_RGBA, 8, settings)
|
result = s.decodePNG(LCT_RGBA, 8, settings)
|
||||||
except:
|
except PNGError, IOError, NZError:
|
||||||
debugEcho getCurrentExceptionMsg()
|
debugEcho getCurrentExceptionMsg()
|
||||||
result = nil
|
result = nil
|
||||||
|
|
||||||
@ -1833,7 +1840,7 @@ proc decodePNG24*(input: string, settings = PNGDecoder(nil)): PNGResult =
|
|||||||
var s = newStringStream(input)
|
var s = newStringStream(input)
|
||||||
if s == nil: return nil
|
if s == nil: return nil
|
||||||
result = s.decodePNG(LCT_RGB, 8, settings)
|
result = s.decodePNG(LCT_RGB, 8, settings)
|
||||||
except:
|
except PNGError, IOError, NZError:
|
||||||
debugEcho getCurrentExceptionMsg()
|
debugEcho getCurrentExceptionMsg()
|
||||||
result = nil
|
result = nil
|
||||||
|
|
||||||
@ -2053,7 +2060,7 @@ method writeChunk(chunk: PNGTrans, png: PNG): bool =
|
|||||||
chunk.writeInt16(chunk.keyG)
|
chunk.writeInt16(chunk.keyG)
|
||||||
chunk.writeInt16(chunk.keyB)
|
chunk.writeInt16(chunk.keyB)
|
||||||
else:
|
else:
|
||||||
raise PNGError("tRNS chunk not allowed for other color models")
|
raise PNGFatal("tRNS chunk not allowed for other color models")
|
||||||
result = true
|
result = true
|
||||||
|
|
||||||
method writeChunk(chunk: PNGBackground, png: PNG): bool =
|
method writeChunk(chunk: PNGBackground, png: PNG): bool =
|
||||||
@ -2123,7 +2130,7 @@ method writeChunk(chunk: PNGSPalette, png: PNG): bool =
|
|||||||
#else: estimate += chunk.palette.len * 10
|
#else: estimate += chunk.palette.len * 10
|
||||||
chunk.writeString chunk.paletteName
|
chunk.writeString chunk.paletteName
|
||||||
chunk.writeByte 0 #null separator
|
chunk.writeByte 0 #null separator
|
||||||
if chunk.sampleDepth notin {8, 16}: raise PNGError("palette sample depth error")
|
if chunk.sampleDepth notin {8, 16}: raise PNGFatal("palette sample depth error")
|
||||||
chunk.writeByte chunk.sampleDepth
|
chunk.writeByte chunk.sampleDepth
|
||||||
|
|
||||||
if chunk.sampleDepth == 8:
|
if chunk.sampleDepth == 8:
|
||||||
@ -2462,7 +2469,7 @@ proc filter[T](output: var openArray[T], input: openArray[T], w, h: int, modeOut
|
|||||||
(modeOut.colorType == LCT_PALETTE or modeOut.bitDepth < 8): strategy = LFS_ZERO
|
(modeOut.colorType == LCT_PALETTE or modeOut.bitDepth < 8): strategy = LFS_ZERO
|
||||||
|
|
||||||
if bpp == 0:
|
if bpp == 0:
|
||||||
raise PNGError("invalid color type")
|
raise PNGFatal("invalid color type")
|
||||||
|
|
||||||
case strategy
|
case strategy
|
||||||
of LFS_ZERO: filterZero(output, input, w, h, bpp)
|
of LFS_ZERO: filterZero(output, input, w, h, bpp)
|
||||||
@ -2690,31 +2697,31 @@ proc encoderCore(png: PNG) =
|
|||||||
var sequenceNumber = 0
|
var sequenceNumber = 0
|
||||||
|
|
||||||
if not bitDepthAllowed(modeIn.colorType, modeIn.bitDepth):
|
if not bitDepthAllowed(modeIn.colorType, modeIn.bitDepth):
|
||||||
raise PNGError("modeIn colorType and bitDepth combination not allowed")
|
raise PNGFatal("modeIn colorType and bitDepth combination not allowed")
|
||||||
|
|
||||||
if not bitDepthAllowed(modeOut.colorType, modeOut.bitDepth):
|
if not bitDepthAllowed(modeOut.colorType, modeOut.bitDepth):
|
||||||
raise PNGError("modeOut colorType and bitDepth combination not allowed")
|
raise PNGFatal("modeOut colorType and bitDepth combination not allowed")
|
||||||
|
|
||||||
if(modeOut.colorType == LCT_PALETTE or state.forcePalette) and
|
if(modeOut.colorType == LCT_PALETTE or state.forcePalette) and
|
||||||
(modeOut.paletteSize == 0 or modeOut.paletteSize > 256):
|
(modeOut.paletteSize == 0 or modeOut.paletteSize > 256):
|
||||||
raise PNGError("invalid palette size, it is only allowed to be 1-256")
|
raise PNGFatal("invalid palette size, it is only allowed to be 1-256")
|
||||||
|
|
||||||
if state.filterStrategy == LFS_PREDEFINED:
|
if state.filterStrategy == LFS_PREDEFINED:
|
||||||
if state.predefinedFilters.len < png.width:
|
if state.predefinedFilters.len < png.width:
|
||||||
raise PNGError("predefinedFilters contains not enough filterType compared to image height")
|
raise PNGFatal("predefinedFilters contains not enough filterType compared to image height")
|
||||||
|
|
||||||
let inputSize = getRawSize(png.width, png.height, modeIn)
|
let inputSize = getRawSize(png.width, png.height, modeIn)
|
||||||
if png.pixels.len < inputSize:
|
if png.pixels.len < inputSize:
|
||||||
raise PNGError("not enough input to encode")
|
raise PNGFatal("not enough input to encode")
|
||||||
|
|
||||||
if state.autoConvert:
|
if state.autoConvert:
|
||||||
png.autoChooseColor(modeOut, modeIn)
|
png.autoChooseColor(modeOut, modeIn)
|
||||||
|
|
||||||
if state.interlaceMethod notin {IM_NONE, IM_INTERLACED}:
|
if state.interlaceMethod notin {IM_NONE, IM_INTERLACED}:
|
||||||
raise PNGError("unexisting interlace mode")
|
raise PNGFatal("unexisting interlace mode")
|
||||||
|
|
||||||
if not bitDepthAllowed(modeOut.colorType, modeOut.bitDepth):
|
if not bitDepthAllowed(modeOut.colorType, modeOut.bitDepth):
|
||||||
raise PNGError("colorType and bitDepth combination not allowed")
|
raise PNGFatal("colorType and bitDepth combination not allowed")
|
||||||
|
|
||||||
if not png.isAPNG: png.apngPixels = @[""]
|
if not png.isAPNG: png.apngPixels = @[""]
|
||||||
shallowCopy(png.apngPixels[0], png.pixels)
|
shallowCopy(png.apngPixels[0], png.pixels)
|
||||||
@ -2747,9 +2754,9 @@ proc encoderCore(png: PNG) =
|
|||||||
|
|
||||||
if png.isAPNG:
|
if png.isAPNG:
|
||||||
if png.apngPixels.len != png.apngChunks.len:
|
if png.apngPixels.len != png.apngChunks.len:
|
||||||
raise PNGError("APNG encoder frame error")
|
raise PNGFatal("APNG encoder frame error")
|
||||||
if png.apngPixels.len == 0:
|
if png.apngPixels.len == 0:
|
||||||
raise PNGError("APNG encoder no frame")
|
raise PNGFatal("APNG encoder no frame")
|
||||||
png.addChunkacTL(png.apngPixels.len, state.numPlays)
|
png.addChunkacTL(png.apngPixels.len, state.numPlays)
|
||||||
if png.firstFrameIsDefaultImage:
|
if png.firstFrameIsDefaultImage:
|
||||||
png.addChunkfcTL(APNGFrameControl(png.apngChunks[0]), sequenceNumber)
|
png.addChunkfcTL(APNGFrameControl(png.apngChunks[0]), sequenceNumber)
|
||||||
@ -2802,9 +2809,9 @@ proc encodePNG*(input: string, w, h: int, settings = PNGEncoder(nil)): PNG =
|
|||||||
png.encoderCore()
|
png.encoderCore()
|
||||||
result = png
|
result = png
|
||||||
|
|
||||||
proc encodePNG*(input: string, colorType: PNGcolorType, bitDepth, w, h: int, settings = PNGEncoder(nil)): PNG =
|
proc encodePNG*(input: string, colorType: PNGColorType, bitDepth, w, h: int, settings = PNGEncoder(nil)): PNG =
|
||||||
if not bitDepthAllowed(colorType, bitDepth):
|
if not bitDepthAllowed(colorType, bitDepth):
|
||||||
raise PNGError("colorType and bitDepth combination not allowed")
|
raise PNGFatal("colorType and bitDepth combination not allowed")
|
||||||
|
|
||||||
var state: PNGEncoder
|
var state: PNGEncoder
|
||||||
if settings == nil: state = makePNGEncoder()
|
if settings == nil: state = makePNGEncoder()
|
||||||
@ -2824,8 +2831,8 @@ proc writeChunks*(png: PNG, s: Stream) =
|
|||||||
s.write PNGSignature
|
s.write PNGSignature
|
||||||
|
|
||||||
for chunk in png.chunks:
|
for chunk in png.chunks:
|
||||||
if not chunk.validateChunk(png): raise PNGError("combine chunk validation error")
|
if not chunk.validateChunk(png): raise PNGFatal("combine chunk validation error")
|
||||||
if not chunk.writeChunk(png): raise PNGError("combine chunk write error")
|
if not chunk.writeChunk(png): raise PNGFatal("combine chunk write error")
|
||||||
chunk.length = chunk.data.len
|
chunk.length = chunk.data.len
|
||||||
chunk.crc = crc32(crc32(0, $chunk.chunkType), chunk.data)
|
chunk.crc = crc32(crc32(0, $chunk.chunkType), chunk.data)
|
||||||
|
|
||||||
@ -2835,14 +2842,14 @@ proc writeChunks*(png: PNG, s: Stream) =
|
|||||||
s.writeInt32BE cast[int](chunk.crc)
|
s.writeInt32BE cast[int](chunk.crc)
|
||||||
|
|
||||||
when not defined(js):
|
when not defined(js):
|
||||||
proc savePNG*(fileName, input: string, colorType: PNGcolorType, bitDepth, w, h: int): bool =
|
proc savePNG*(fileName, input: string, colorType: PNGColorType, bitDepth, w, h: int): bool =
|
||||||
try:
|
try:
|
||||||
var png = encodePNG(input, colorType, bitDepth, w, h)
|
var png = encodePNG(input, colorType, bitDepth, w, h)
|
||||||
var s = newFileStream(fileName, fmWrite)
|
var s = newFileStream(fileName, fmWrite)
|
||||||
png.writeChunks s
|
png.writeChunks s
|
||||||
s.close()
|
s.close()
|
||||||
result = true
|
result = true
|
||||||
except:
|
except PNGError, IOError, NZError:
|
||||||
debugEcho getCurrentExceptionMsg()
|
debugEcho getCurrentExceptionMsg()
|
||||||
result = false
|
result = false
|
||||||
|
|
||||||
@ -2852,7 +2859,7 @@ when not defined(js):
|
|||||||
proc savePNG24*(fileName, input: string, w, h: int): bool =
|
proc savePNG24*(fileName, input: string, w, h: int): bool =
|
||||||
result = savePNG(fileName, input, LCT_RGB, 8, w, h)
|
result = savePNG(fileName, input, LCT_RGB, 8, w, h)
|
||||||
|
|
||||||
proc prepareAPNG*(colorType: PNGcolorType, bitDepth, numPlays: int, settings = PNGEncoder(nil)): PNG =
|
proc prepareAPNG*(colorType: PNGColorType, bitDepth, numPlays: int, settings = PNGEncoder(nil)): PNG =
|
||||||
var state: PNGEncoder
|
var state: PNGEncoder
|
||||||
if settings == nil: state = makePNGEncoder()
|
if settings == nil: state = makePNGEncoder()
|
||||||
else: state = settings
|
else: state = settings
|
||||||
@ -2922,7 +2929,7 @@ proc encodeAPNG*(png: PNG): string =
|
|||||||
var s = newStringStream()
|
var s = newStringStream()
|
||||||
png.writeChunks s
|
png.writeChunks s
|
||||||
result = s.data
|
result = s.data
|
||||||
except:
|
except PNGError, IOError, NZError:
|
||||||
debugEcho getCurrentExceptionMsg()
|
debugEcho getCurrentExceptionMsg()
|
||||||
result = ""
|
result = ""
|
||||||
|
|
||||||
@ -2934,7 +2941,7 @@ when not defined(js):
|
|||||||
png.writeChunks s
|
png.writeChunks s
|
||||||
s.close()
|
s.close()
|
||||||
result = true
|
result = true
|
||||||
except:
|
except PNGError, IOError, NZError:
|
||||||
debugEcho getCurrentExceptionMsg()
|
debugEcho getCurrentExceptionMsg()
|
||||||
result = false
|
result = false
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
# Package
|
# Package
|
||||||
version = "0.2.6"
|
version = "0.2.7"
|
||||||
author = "Andri Lim"
|
author = "Andri Lim"
|
||||||
description = "PNG encoder and decoder"
|
description = "PNG encoder and decoder"
|
||||||
license = "MIT"
|
license = "MIT"
|
||||||
|
@ -58,7 +58,7 @@ type
|
|||||||
data: string
|
data: string
|
||||||
databitlen: int
|
databitlen: int
|
||||||
|
|
||||||
NZError = ref object of Exception
|
NZError* = object of CatchableError
|
||||||
|
|
||||||
NZHash = object
|
NZHash = object
|
||||||
head: seq[int] #hash value to head circular pos
|
head: seq[int] #hash value to head circular pos
|
||||||
@ -133,9 +133,8 @@ type
|
|||||||
mode: nzStreamMode
|
mode: nzStreamMode
|
||||||
ignoreAdler32*: bool
|
ignoreAdler32*: bool
|
||||||
|
|
||||||
proc newNZError(msg: string): NZError =
|
template NZFatal(msg: string): untyped =
|
||||||
new(result)
|
newException(NZError, msg)
|
||||||
result.msg = msg
|
|
||||||
|
|
||||||
proc readBit(s: BitStream): int {.inline.} =
|
proc readBit(s: BitStream): int {.inline.} =
|
||||||
result = (ord(s.data[s.bitpointer shr 3]) shr (s.bitpointer and 0x07)) and 0x01
|
result = (ord(s.data[s.bitpointer shr 3]) shr (s.bitpointer and 0x07)) and 0x01
|
||||||
@ -151,7 +150,7 @@ proc readBitsFromStream(s: var BitStream, nbits: int): int =
|
|||||||
|
|
||||||
proc readBitsSafe(s: var BitStream, nbits: int): int =
|
proc readBitsSafe(s: var BitStream, nbits: int): int =
|
||||||
if s.bitpointer + nbits > s.databitlen:
|
if s.bitpointer + nbits > s.databitlen:
|
||||||
raise newNZError("bit pointer jumps past memory")
|
raise NZFatal("bit pointer jumps past memory")
|
||||||
|
|
||||||
for i in 0..nbits-1:
|
for i in 0..nbits-1:
|
||||||
inc(result, s.readBit shl i)
|
inc(result, s.readBit shl i)
|
||||||
@ -181,7 +180,7 @@ proc HuffmanTree_make2DTree(tree: var HuffmanTree) =
|
|||||||
let branch = 2 * treepos + bit
|
let branch = 2 * treepos + bit
|
||||||
#oversubscribed, see comment in lodepng_error_text
|
#oversubscribed, see comment in lodepng_error_text
|
||||||
if treepos > 2147483647 or treepos + 2 > tree.numcodes:
|
if treepos > 2147483647 or treepos + 2 > tree.numcodes:
|
||||||
raise newNZError("oversubscribed")
|
raise NZFatal("oversubscribed")
|
||||||
|
|
||||||
if tree.tree2d[branch] != 32767: #not yet filled in
|
if tree.tree2d[branch] != 32767: #not yet filled in
|
||||||
treepos = tree.tree2d[branch] - tree.numcodes
|
treepos = tree.tree2d[branch] - tree.numcodes
|
||||||
@ -301,7 +300,7 @@ proc huffman_code_lengths(frequencies: openarray[int], numcodes, maxbitlen: int)
|
|||||||
coinmem, numcoins: int
|
coinmem, numcoins: int
|
||||||
|
|
||||||
if numcodes == 0:
|
if numcodes == 0:
|
||||||
raise newNZError("a tree of 0 symbols is not supposed to be made")
|
raise NZFatal("a tree of 0 symbols is not supposed to be made")
|
||||||
|
|
||||||
for i in 0..numcodes-1:
|
for i in 0..numcodes-1:
|
||||||
if frequencies[i] > 0:
|
if frequencies[i] > 0:
|
||||||
@ -410,7 +409,7 @@ proc readInt16(s: var BitStream): int =
|
|||||||
#go to first boundary of byte
|
#go to first boundary of byte
|
||||||
while (s.bitpointer and 0x7) != 0: inc s.bitpointer
|
while (s.bitpointer and 0x7) != 0: inc s.bitpointer
|
||||||
var p = s.bitpointer div 8 #byte position
|
var p = s.bitpointer div 8 #byte position
|
||||||
if p + 2 > s.data.len: raise newNZError("bit pointer will jump past memory")
|
if p + 2 > s.data.len: raise NZFatal("bit pointer will jump past memory")
|
||||||
result = ord(s.data[p]) + 256 * ord(s.data[p + 1])
|
result = ord(s.data[p]) + 256 * ord(s.data[p + 1])
|
||||||
inc(s.bitpointer, 16)
|
inc(s.bitpointer, 16)
|
||||||
|
|
||||||
@ -420,7 +419,7 @@ proc getBytePosition(s: var BitStream): int =
|
|||||||
proc readByte(s: var BitStream): int =
|
proc readByte(s: var BitStream): int =
|
||||||
while (s.bitpointer and 0x7) != 0: inc s.bitpointer
|
while (s.bitpointer and 0x7) != 0: inc s.bitpointer
|
||||||
var p = s.bitpointer div 8 #byte position
|
var p = s.bitpointer div 8 #byte position
|
||||||
if p + 1 >= s.data.len: raise newNZError("bit pointer will jump past memory")
|
if p + 1 >= s.data.len: raise NZFatal("bit pointer will jump past memory")
|
||||||
result = ord(s.data[p])
|
result = ord(s.data[p])
|
||||||
inc(s.bitpointer, 8)
|
inc(s.bitpointer, 8)
|
||||||
|
|
||||||
@ -433,12 +432,12 @@ proc inflateNoCompression(nz: nzStream) =
|
|||||||
|
|
||||||
#check if 16-bit NLEN is really the one's complement of LEN
|
#check if 16-bit NLEN is really the one's complement of LEN
|
||||||
if LEN + NLEN != 65535:
|
if LEN + NLEN != 65535:
|
||||||
raise newNZError("NLEN is not one's complement of LEN")
|
raise NZFatal("NLEN is not one's complement of LEN")
|
||||||
|
|
||||||
#read the literal data: LEN bytes are now stored in the out buffer
|
#read the literal data: LEN bytes are now stored in the out buffer
|
||||||
var p = nz.bits.getBytePosition
|
var p = nz.bits.getBytePosition
|
||||||
if p + LEN > inlength:
|
if p + LEN > inlength:
|
||||||
raise newNZError("reading outside of input buffer")
|
raise NZFatal("reading outside of input buffer")
|
||||||
|
|
||||||
var pos = nz.data.len
|
var pos = nz.data.len
|
||||||
nz.data.setLen(pos + LEN)
|
nz.data.setLen(pos + LEN)
|
||||||
@ -495,7 +494,7 @@ proc getTreeInflateDynamic(s: var BitStream, tree_ll, tree_d: var HuffmanTree) =
|
|||||||
var tree_cl: HuffmanTree
|
var tree_cl: HuffmanTree
|
||||||
|
|
||||||
if s.bitpointer + 14 > inbitlength:
|
if s.bitpointer + 14 > inbitlength:
|
||||||
raise newNZError("the bit pointer is or will go past the memory")
|
raise NZFatal("the bit pointer is or will go past the memory")
|
||||||
|
|
||||||
#number of literal/length codes + 257.
|
#number of literal/length codes + 257.
|
||||||
#Unlike the spec, the value 257 is added to it here already
|
#Unlike the spec, the value 257 is added to it here already
|
||||||
@ -509,7 +508,7 @@ proc getTreeInflateDynamic(s: var BitStream, tree_ll, tree_d: var HuffmanTree) =
|
|||||||
let HCLEN = s.readBitsFromStream(4) + 4
|
let HCLEN = s.readBitsFromStream(4) + 4
|
||||||
|
|
||||||
if s.bitpointer + HCLEN * 3 > inbitlength:
|
if s.bitpointer + HCLEN * 3 > inbitlength:
|
||||||
raise newNZError("the bit pointer is or will go past the memory")
|
raise NZFatal("the bit pointer is or will go past the memory")
|
||||||
|
|
||||||
#read the code length codes out of 3 * (amount of code length codes) bits
|
#read the code length codes out of 3 * (amount of code length codes) bits
|
||||||
for i in 0..NUM_CODE_LENGTH_CODES-1:
|
for i in 0..NUM_CODE_LENGTH_CODES-1:
|
||||||
@ -533,7 +532,7 @@ proc getTreeInflateDynamic(s: var BitStream, tree_ll, tree_d: var HuffmanTree) =
|
|||||||
var replength = 3 #read in the 2 bits that indicate repeat length (3-6)
|
var replength = 3 #read in the 2 bits that indicate repeat length (3-6)
|
||||||
var value = 0 #set value to the previous code
|
var value = 0 #set value to the previous code
|
||||||
|
|
||||||
if i == 0: raise newNZError("can't repeat previous if i is 0")
|
if i == 0: raise NZFatal("can't repeat previous if i is 0")
|
||||||
replength += s.readBitsSafe(2)
|
replength += s.readBitsSafe(2)
|
||||||
|
|
||||||
if i < HLIT + 1: value = bitlen_ll[i - 1]
|
if i < HLIT + 1: value = bitlen_ll[i - 1]
|
||||||
@ -541,7 +540,7 @@ proc getTreeInflateDynamic(s: var BitStream, tree_ll, tree_d: var HuffmanTree) =
|
|||||||
|
|
||||||
#repeat this value in the next lengths
|
#repeat this value in the next lengths
|
||||||
for n in 0..replength-1:
|
for n in 0..replength-1:
|
||||||
if i >= HLIT + HDIST: raise newNZError("i is larger than the amount of codes")
|
if i >= HLIT + HDIST: raise NZFatal("i is larger than the amount of codes")
|
||||||
if i < HLIT: bitlen_ll[i] = value
|
if i < HLIT: bitlen_ll[i] = value
|
||||||
else: bitlen_d[i - HLIT] = value
|
else: bitlen_d[i - HLIT] = value
|
||||||
inc(i)
|
inc(i)
|
||||||
@ -551,7 +550,7 @@ proc getTreeInflateDynamic(s: var BitStream, tree_ll, tree_d: var HuffmanTree) =
|
|||||||
|
|
||||||
#repeat this value in the next lengths
|
#repeat this value in the next lengths
|
||||||
for n in 0..replength-1:
|
for n in 0..replength-1:
|
||||||
if i >= HLIT + HDIST: raise newNZError("i is larger than the amount of codes")
|
if i >= HLIT + HDIST: raise NZFatal("i is larger than the amount of codes")
|
||||||
if i < HLIT: bitlen_ll[i] = 0
|
if i < HLIT: bitlen_ll[i] = 0
|
||||||
else: bitlen_d[i - HLIT] = 0
|
else: bitlen_d[i - HLIT] = 0
|
||||||
inc(i)
|
inc(i)
|
||||||
@ -561,7 +560,7 @@ proc getTreeInflateDynamic(s: var BitStream, tree_ll, tree_d: var HuffmanTree) =
|
|||||||
|
|
||||||
#repeat this value in the next lengths
|
#repeat this value in the next lengths
|
||||||
for n in 0..replength-1:
|
for n in 0..replength-1:
|
||||||
if i >= HLIT + HDIST: raise newNZError("i is larger than the amount of codes")
|
if i >= HLIT + HDIST: raise NZFatal("i is larger than the amount of codes")
|
||||||
if i < HLIT: bitlen_ll[i] = 0
|
if i < HLIT: bitlen_ll[i] = 0
|
||||||
else: bitlen_d[i - HLIT] = 0
|
else: bitlen_d[i - HLIT] = 0
|
||||||
inc(i)
|
inc(i)
|
||||||
@ -569,14 +568,14 @@ proc getTreeInflateDynamic(s: var BitStream, tree_ll, tree_d: var HuffmanTree) =
|
|||||||
if code == -1:
|
if code == -1:
|
||||||
#return error code 10 or 11 depending on the situation that happened in huffmanDecodeSymbol
|
#return error code 10 or 11 depending on the situation that happened in huffmanDecodeSymbol
|
||||||
#(10=no endcode, 11=wrong jump outside of tree)
|
#(10=no endcode, 11=wrong jump outside of tree)
|
||||||
if s.bitpointer > inbitlength: raise newNZError("no endcode")
|
if s.bitpointer > inbitlength: raise NZFatal("no endcode")
|
||||||
else: raise newNZError("wrong jump outside of tree")
|
else: raise NZFatal("wrong jump outside of tree")
|
||||||
else:
|
else:
|
||||||
raise newNZError("unexisting code, this can never happen")
|
raise NZFatal("unexisting code, this can never happen")
|
||||||
break
|
break
|
||||||
|
|
||||||
if bitlen_ll[256] == 0:
|
if bitlen_ll[256] == 0:
|
||||||
raise newNZError("the length of the end code 256 must be larger than 0")
|
raise NZFatal("the length of the end code 256 must be larger than 0")
|
||||||
|
|
||||||
#now we've finally got HLIT and HDIST,
|
#now we've finally got HLIT and HDIST,
|
||||||
#so generate the code trees, and the function is done
|
#so generate the code trees, and the function is done
|
||||||
@ -613,10 +612,10 @@ proc inflateHuffmanBlock(nz: nzStream, blockType: int) =
|
|||||||
if code_ll == -1: #huffmanDecodeSymbol returns -1 in case of error
|
if code_ll == -1: #huffmanDecodeSymbol returns -1 in case of error
|
||||||
#return error code 10 or 11 depending on the situation that happened in huffmanDecodeSymbol
|
#return error code 10 or 11 depending on the situation that happened in huffmanDecodeSymbol
|
||||||
#(10=no endcode, 11=wrong jump outside of tree)
|
#(10=no endcode, 11=wrong jump outside of tree)
|
||||||
if nz.bits.bitpointer > inbitlength: raise newNZError("no endcode")
|
if nz.bits.bitpointer > inbitlength: raise NZFatal("no endcode")
|
||||||
else: raise newNZError("wrong jump outside of tree")
|
else: raise NZFatal("wrong jump outside of tree")
|
||||||
else:
|
else:
|
||||||
raise newNZError("invalid distance code (30-31 are never used)")
|
raise NZFatal("invalid distance code (30-31 are never used)")
|
||||||
break
|
break
|
||||||
var distance = DISTANCEBASE[code_d]
|
var distance = DISTANCEBASE[code_d]
|
||||||
|
|
||||||
@ -627,7 +626,7 @@ proc inflateHuffmanBlock(nz: nzStream, blockType: int) =
|
|||||||
#part 5: fill in all the out[n] values based on the length and dist
|
#part 5: fill in all the out[n] values based on the length and dist
|
||||||
let start = nz.data.len
|
let start = nz.data.len
|
||||||
if distance > start:
|
if distance > start:
|
||||||
raise newNZError("too long backward distance")
|
raise NZFatal("too long backward distance")
|
||||||
var backward = start - distance
|
var backward = start - distance
|
||||||
|
|
||||||
nz.data.setLen(start + length)
|
nz.data.setLen(start + length)
|
||||||
@ -640,8 +639,8 @@ proc inflateHuffmanBlock(nz: nzStream, blockType: int) =
|
|||||||
else: #if(code == -1) huffmanDecodeSymbol returns -1 in case of error
|
else: #if(code == -1) huffmanDecodeSymbol returns -1 in case of error
|
||||||
#return error code 10 or 11 depending on the situation that happened in huffmanDecodeSymbol
|
#return error code 10 or 11 depending on the situation that happened in huffmanDecodeSymbol
|
||||||
#(10=no endcode, 11=wrong jump outside of tree)
|
#(10=no endcode, 11=wrong jump outside of tree)
|
||||||
if nz.bits.bitpointer > inbitlength: raise newNZError("no endcode")
|
if nz.bits.bitpointer > inbitlength: raise NZFatal("no endcode")
|
||||||
else: raise newNZError("wrong jump outside of tree")
|
else: raise NZFatal("wrong jump outside of tree")
|
||||||
break
|
break
|
||||||
|
|
||||||
proc nzInflate(nz: nzStream) =
|
proc nzInflate(nz: nzStream) =
|
||||||
@ -655,7 +654,7 @@ proc nzInflate(nz: nzStream) =
|
|||||||
finalBlock = nz.bits.readBitFromStream != 0
|
finalBlock = nz.bits.readBitFromStream != 0
|
||||||
let blockType = nz.bits.readBitFromStream + 2 * nz.bits.readBitFromStream
|
let blockType = nz.bits.readBitFromStream + 2 * nz.bits.readBitFromStream
|
||||||
|
|
||||||
if blockType == 3: raise newNZError("invalid blockType")
|
if blockType == 3: raise NZFatal("invalid blockType")
|
||||||
elif blockType == 0: nz.inflateNoCompression #no compression
|
elif blockType == 0: nz.inflateNoCompression #no compression
|
||||||
else: nz.inflateHuffmanBlock(blockType) #compression, blockType 01 or 10
|
else: nz.inflateHuffmanBlock(blockType) #compression, blockType 01 or 10
|
||||||
|
|
||||||
@ -858,9 +857,9 @@ proc encodeLZ77(nz: nzStream, hash: var NZHash, inpos, insize: int): seq[int] =
|
|||||||
current_offset, current_length: int
|
current_offset, current_length: int
|
||||||
|
|
||||||
if (nz.windowsize == 0) or (nz.windowsize > 32768):
|
if (nz.windowsize == 0) or (nz.windowsize > 32768):
|
||||||
raise newNZError("windowsize smaller/larger than allowed")
|
raise NZFatal("windowsize smaller/larger than allowed")
|
||||||
if (nz.windowsize and (nz.windowsize - 1)) != 0:
|
if (nz.windowsize and (nz.windowsize - 1)) != 0:
|
||||||
raise newNZError("must be power of two")
|
raise NZFatal("must be power of two")
|
||||||
|
|
||||||
var nicematch = min(nz.nicematch, MAX_SUPPORTED_DEFLATE_LENGTH)
|
var nicematch = min(nz.nicematch, MAX_SUPPORTED_DEFLATE_LENGTH)
|
||||||
var pos = inpos
|
var pos = inpos
|
||||||
@ -940,7 +939,7 @@ proc encodeLZ77(nz: nzStream, hash: var NZHash, inpos, insize: int): seq[int] =
|
|||||||
|
|
||||||
if lazy != 0:
|
if lazy != 0:
|
||||||
lazy = 0
|
lazy = 0
|
||||||
if pos == 0: raise newNZError("lazy matching at pos 0 is impossible")
|
if pos == 0: raise NZFatal("lazy matching at pos 0 is impossible")
|
||||||
if length > lazylength + 1:
|
if length > lazylength + 1:
|
||||||
#push the previous character as literal
|
#push the previous character as literal
|
||||||
result.add ord(nz.data[pos - 1])
|
result.add ord(nz.data[pos - 1])
|
||||||
@ -952,7 +951,7 @@ proc encodeLZ77(nz: nzStream, hash: var NZHash, inpos, insize: int): seq[int] =
|
|||||||
dec pos
|
dec pos
|
||||||
|
|
||||||
if(length >= 3) and (offset > nz.windowsize):
|
if(length >= 3) and (offset > nz.windowsize):
|
||||||
raise newNZError("too big (or overflown negative) offset")
|
raise NZFatal("too big (or overflown negative) offset")
|
||||||
|
|
||||||
#encode it as length/distance pair or literal value
|
#encode it as length/distance pair or literal value
|
||||||
if length < 3: #only lengths of 3 or higher are supported as length/distance pair
|
if length < 3: #only lengths of 3 or higher are supported as length/distance pair
|
||||||
@ -1169,7 +1168,7 @@ proc deflateDynamic(nz: nzStream, hash: var NZHash, datapos, dataend: int, final
|
|||||||
nz.bits.writeLZ77data(lz77, tree_ll, tree_d)
|
nz.bits.writeLZ77data(lz77, tree_ll, tree_d)
|
||||||
|
|
||||||
if HuffmanTree_getLength(tree_ll, 256) == 0:
|
if HuffmanTree_getLength(tree_ll, 256) == 0:
|
||||||
raise newNZError("the length of the end code 256 must be larger than 0")
|
raise NZFatal("the length of the end code 256 must be larger than 0")
|
||||||
|
|
||||||
#write the end code
|
#write the end code
|
||||||
nz.bits.addHuffmanSymbol(tree_ll, 256)
|
nz.bits.addHuffmanSymbol(tree_ll, 256)
|
||||||
@ -1179,7 +1178,7 @@ proc nzDeflate(nz: nzStream) =
|
|||||||
var blocksize = 0
|
var blocksize = 0
|
||||||
var insize = nz.data.len
|
var insize = nz.data.len
|
||||||
|
|
||||||
if nz.btype > 2: raise newNZError("invalid block type")
|
if nz.btype > 2: raise NZFatal("invalid block type")
|
||||||
elif nz.btype == 0:
|
elif nz.btype == 0:
|
||||||
nz.deflateNoCompression
|
nz.deflateNoCompression
|
||||||
return
|
return
|
||||||
@ -1308,14 +1307,14 @@ proc readInt32(input: string): uint32 =
|
|||||||
proc zlib_decompress*(nz: nzStream): string =
|
proc zlib_decompress*(nz: nzStream): string =
|
||||||
let insize = nz.bits.data.len
|
let insize = nz.bits.data.len
|
||||||
|
|
||||||
if insize < 2: raise newNZError("size of zlib data too small")
|
if insize < 2: raise NZFatal("size of zlib data too small")
|
||||||
|
|
||||||
#read information from zlib header
|
#read information from zlib header
|
||||||
let CMF = nz.bits.readByte
|
let CMF = nz.bits.readByte
|
||||||
let FLG = nz.bits.readByte
|
let FLG = nz.bits.readByte
|
||||||
|
|
||||||
if ((CMF * 256 + FLG) mod 31) != 0:
|
if ((CMF * 256 + FLG) mod 31) != 0:
|
||||||
raise newNZError(" zlib header must be a multiple of 31")
|
raise NZFatal(" zlib header must be a multiple of 31")
|
||||||
#the FCHECK value is supposed to be made that way
|
#the FCHECK value is supposed to be made that way
|
||||||
|
|
||||||
#let CM = CMF and 15
|
#let CM = CMF and 15
|
||||||
@ -1334,12 +1333,13 @@ proc zlib_decompress*(nz: nzStream): string =
|
|||||||
|
|
||||||
let checksum = nz.bits.data.substr(insize-4, insize-1).readInt32
|
let checksum = nz.bits.data.substr(insize-4, insize-1).readInt32
|
||||||
nz.bits.data.setLen(insize-4)
|
nz.bits.data.setLen(insize-4)
|
||||||
|
nz.bits.databitlen = (insize-4) * 8 # reset databitlen
|
||||||
|
|
||||||
nz.nzInflate
|
nz.nzInflate
|
||||||
|
|
||||||
if not nz.ignoreAdler32:
|
if not nz.ignoreAdler32:
|
||||||
let adler32 = nzAdler32(1, nz.data)
|
let adler32 = nzAdler32(1, nz.data)
|
||||||
if checksum != adler32:
|
if checksum != adler32:
|
||||||
raise newNZError("adler checksum not correct, data must be corrupted")
|
raise NZFatal("adler checksum not correct, data must be corrupted")
|
||||||
|
|
||||||
result = nz.nzGetResult
|
result = nz.nzGetResult
|
||||||
|
@ -3,4 +3,5 @@ import
|
|||||||
test_codec,
|
test_codec,
|
||||||
test_suite,
|
test_suite,
|
||||||
test_nimz,
|
test_nimz,
|
||||||
test_filters
|
test_filters,
|
||||||
|
test_crash
|
||||||
|
9
tests/fuzz_codec.nim
Normal file
9
tests/fuzz_codec.nim
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
import testutils/fuzzing, ../nimPNG
|
||||||
|
|
||||||
|
proc toString(x: openArray[byte]): string =
|
||||||
|
result = newString(x.len)
|
||||||
|
if x.len != 0:
|
||||||
|
copyMem(result[0].addr, x[0].unsafeAddr, x.len)
|
||||||
|
|
||||||
|
test:
|
||||||
|
let png = decodePNG32(toString(payload))
|
BIN
tests/invalidInput/crash-1
Normal file
BIN
tests/invalidInput/crash-1
Normal file
Binary file not shown.
After Width: | Height: | Size: 145 B |
BIN
tests/invalidInput/crash-2
Normal file
BIN
tests/invalidInput/crash-2
Normal file
Binary file not shown.
After Width: | Height: | Size: 377 B |
BIN
tests/invalidInput/crash-3
Normal file
BIN
tests/invalidInput/crash-3
Normal file
Binary file not shown.
After Width: | Height: | Size: 245 B |
BIN
tests/invalidInput/crash-4
Normal file
BIN
tests/invalidInput/crash-4
Normal file
Binary file not shown.
After Width: | Height: | Size: 2.0 KiB |
BIN
tests/invalidInput/crash-5
Normal file
BIN
tests/invalidInput/crash-5
Normal file
Binary file not shown.
After Width: | Height: | Size: 971 B |
11
tests/test_crash.nim
Normal file
11
tests/test_crash.nim
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
import ../nimPNG, unittest, os
|
||||||
|
|
||||||
|
proc main() =
|
||||||
|
suite "parse invalid input":
|
||||||
|
for x in walkDirRec("tests" / "invalidInput"):
|
||||||
|
let y = splitPath(x)
|
||||||
|
test y.tail:
|
||||||
|
discard loadPNG32(x)
|
||||||
|
check true
|
||||||
|
|
||||||
|
main()
|
Loading…
x
Reference in New Issue
Block a user