Merge pull request #8 from guzba/master

bestspeed png encode + a bit of jpg
This commit is contained in:
treeform 2020-11-22 20:03:53 -08:00 committed by GitHub
commit 733687348e
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
19 changed files with 82 additions and 9 deletions

View file

@ -12,6 +12,7 @@ type
Jpg = object Jpg = object
width, height: int width, height: int
components: array[3, Component] components: array[3, Component]
quantizationTables: array[4, array[64, uint8]]
template failInvalid() = template failInvalid() =
raise newException(PixieError, "Invalid JPG buffer, unable to load") raise newException(PixieError, "Invalid JPG buffer, unable to load")
@ -71,10 +72,59 @@ proc decodeSOF(jpg: var Jpg, data: seq[uint8], pos: var int) =
pos += 3 pos += 3
proc decodeDHT(data: seq[uint8], pos: var int) = proc decodeDHT(data: seq[uint8], pos: var int) =
skipSegment(data, pos) # skipSegment(data, pos)
# debugEcho pos
let
segmentLen = readSegmentLen(data, pos)
stop = pos + segmentLen
pos += 2
proc decodeDQT(data: seq[uint8], pos: var int) = while stop - pos >= 17:
skipSegment(data, pos) let info = data[pos]
if (info and 0b11100000) != 0:
failInvalid()
var counts: array[17, int]
for codeLen in 1 .. 16:
counts[codeLen] = data[pos + codeLen].int
debugEcho counts
pos += 17
for codeLen in 1 .. 16:
discard
break
pos = stop
proc decodeDQT(jpg: var Jpg, data: seq[uint8], pos: var int) =
let
segmentLen = readSegmentLen(data, pos)
stop = pos + segmentLen
pos += 2
while stop - pos >= 65:
let
info = data[pos]
qt = info and 0b00001111
precision = info and 0b11110000
if qt > 3:
failInvalid()
if precision != 0:
raise newException(
PixieError, "Unsuppored JPG qantization table precision"
)
inc pos
for i in 0 ..< 64:
jpg.quantizationTables[qt][i] = data[pos + i]
pos += 64
proc decodeSOS(data: seq[uint8], pos: var int) = proc decodeSOS(data: seq[uint8], pos: var int) =
let segmentLen = readSegmentLen(data, pos) let segmentLen = readSegmentLen(data, pos)
@ -144,7 +194,7 @@ proc decodeJpg*(data: seq[uint8]): Image =
of 0xC4: # Define Huffman Tables of 0xC4: # Define Huffman Tables
decodeDHT(data, pos) decodeDHT(data, pos)
of 0xDB: # Define Quantanization Table(s) of 0xDB: # Define Quantanization Table(s)
decodeDQT(data, pos) jpg.decodeDQT(data, pos)
# of 0xDD: # Define Restart Interval # of 0xDD: # Define Restart Interval
of 0xDA: # Start of Scan of 0xDA: # Start of Scan
decodeSOS(data, pos) decodeSOS(data, pos)

View file

@ -469,7 +469,7 @@ proc encodePng*(
let compressed = let compressed =
try: try:
compress(filtered, DefaultCompression, dfZlib) compress(filtered, BestSpeed, dfZlib)
except ZippyError: except ZippyError:
raise newException( raise newException(
PixieError, "Unexpected error compressing PNG image data" PixieError, "Unexpected error compressing PNG image data"

View file

@ -1,16 +1,27 @@
import pixie/fileformats/png, stb_image/read as stbi, fidget/opengl/perf, nimPNG import pixie/fileformats/png, stb_image/read as stbi, stb_image/write as stbr,
fidget/opengl/perf, nimPNG
let data = readFile("tests/data/lenna.png") let data = readFile("tests/data/lenna.png")
timeIt "pixie": timeIt "pixie decode":
for i in 0 ..< 100: for i in 0 ..< 100:
discard decodePng(cast[seq[uint8]](data)) discard decodePng(cast[seq[uint8]](data))
timeIt "nimPNG": timeIt "pixie encode":
let decoded = decodePng(cast[seq[uint8]](data))
for i in 0 ..< 1:
discard encodePng(decoded).len
timeIt "nimPNG decode":
for i in 0 ..< 100: for i in 0 ..< 100:
discard decodePNG32(data) discard decodePNG32(data)
timeIt "stb_image": timeIt "nimPNG encode":
let decoded = decodePNG32(data)
for i in 0 ..< 100:
discard encodePNG32(decoded.data, decoded.width, decoded.height).pixels.len
timeIt "stb_image decode":
for i in 0 ..< 100: for i in 0 ..< 100:
var width, height, channels: int var width, height, channels: int
discard loadFromMemory( discard loadFromMemory(
@ -20,3 +31,15 @@ timeIt "stb_image":
channels, channels,
stbi.RGBA stbi.RGBA
) )
timeIt "stb_image encode":
var width, height, channels: int
let decoded = loadFromMemory(
cast[seq[byte]](data),
width,
height,
channels,
stbi.RGBA
)
for i in 0 ..< 100:
discard writePNG(width, height, channels, decoded).len

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.5 KiB

After

Width:  |  Height:  |  Size: 1.5 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.6 KiB

After

Width:  |  Height:  |  Size: 1.7 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 328 B

After

Width:  |  Height:  |  Size: 342 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 352 B

After

Width:  |  Height:  |  Size: 370 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 328 B

After

Width:  |  Height:  |  Size: 342 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 343 B

After

Width:  |  Height:  |  Size: 359 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.5 KiB

After

Width:  |  Height:  |  Size: 1.5 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.6 KiB

After

Width:  |  Height:  |  Size: 1.7 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 328 B

After

Width:  |  Height:  |  Size: 342 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 352 B

After

Width:  |  Height:  |  Size: 370 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 328 B

After

Width:  |  Height:  |  Size: 342 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 343 B

After

Width:  |  Height:  |  Size: 359 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.5 KiB

After

Width:  |  Height:  |  Size: 1.5 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.6 KiB

After

Width:  |  Height:  |  Size: 1.7 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.5 KiB

After

Width:  |  Height:  |  Size: 1.5 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.6 KiB

After

Width:  |  Height:  |  Size: 1.7 KiB