tmp jpeg fix
This commit is contained in:
parent
0ad674fc29
commit
fd1b02dd24
1 changed files with 92 additions and 66 deletions
|
@ -59,7 +59,7 @@ type
|
||||||
widthCoeff, heightCoeff: int
|
widthCoeff, heightCoeff: int
|
||||||
coeff, lineBuf: seq[uint8]
|
coeff, lineBuf: seq[uint8]
|
||||||
blocks: seq[seq[array[64, int16]]]
|
blocks: seq[seq[array[64, int16]]]
|
||||||
channel: seq[uint8]
|
channel: Mask
|
||||||
|
|
||||||
DecoderState = object
|
DecoderState = object
|
||||||
buffer: ptr UncheckedArray[uint8]
|
buffer: ptr UncheckedArray[uint8]
|
||||||
|
@ -84,9 +84,46 @@ type
|
||||||
eobRun: int
|
eobRun: int
|
||||||
hitEnd: bool
|
hitEnd: bool
|
||||||
|
|
||||||
|
Mask = ref object
|
||||||
|
## Mask object that holds mask opacity data.
|
||||||
|
width*, height*: int
|
||||||
|
data*: seq[uint8]
|
||||||
|
|
||||||
|
UnsafeMask = distinct Mask
|
||||||
|
|
||||||
when defined(release):
|
when defined(release):
|
||||||
{.push checks: off.}
|
{.push checks: off.}
|
||||||
|
|
||||||
|
proc newMask(width, height: int): Mask {.raises: [PixieError].} =
|
||||||
|
## Creates a new mask with the parameter dimensions.
|
||||||
|
if width <= 0 or height <= 0:
|
||||||
|
raise newException(PixieError, "Mask width and height must be > 0")
|
||||||
|
|
||||||
|
result = Mask()
|
||||||
|
result.width = width
|
||||||
|
result.height = height
|
||||||
|
result.data = newSeq[uint8](width * height)
|
||||||
|
|
||||||
|
template dataIndex(mask: Mask, x, y: int): int =
|
||||||
|
mask.width * y + x
|
||||||
|
|
||||||
|
template unsafe(src: Mask): UnsafeMask =
|
||||||
|
cast[UnsafeMask](src)
|
||||||
|
|
||||||
|
template `[]`(view: UnsafeMask, x, y: int): uint8 =
|
||||||
|
## Gets a value from (x, y) coordinates.
|
||||||
|
## * No bounds checking *
|
||||||
|
## Make sure that x, y are in bounds.
|
||||||
|
## Failure in the assumptions will case unsafe memory reads.
|
||||||
|
cast[Mask](view).data[cast[Mask](view).dataIndex(x, y)]
|
||||||
|
|
||||||
|
template `[]=`(view: UnsafeMask, x, y: int, color: uint8) =
|
||||||
|
## Sets a value from (x, y) coordinates.
|
||||||
|
## * No bounds checking *
|
||||||
|
## Make sure that x, y are in bounds.
|
||||||
|
## Failure in the assumptions will case unsafe memory writes.
|
||||||
|
cast[Mask](view).data[cast[Mask](view).dataIndex(x, y)] = color
|
||||||
|
|
||||||
template failInvalid(reason = "unable to load") =
|
template failInvalid(reason = "unable to load") =
|
||||||
## Throw exception with a reason.
|
## Throw exception with a reason.
|
||||||
raise newException(PixieError, "Invalid JPEG, " & reason)
|
raise newException(PixieError, "Invalid JPEG, " & reason)
|
||||||
|
@ -321,8 +358,7 @@ proc decodeSOF0(state: var DecoderState) =
|
||||||
|
|
||||||
component.widthStride = state.numMcuWide * component.yScale * 8
|
component.widthStride = state.numMcuWide * component.yScale * 8
|
||||||
component.heightStride = state.numMcuHigh * component.xScale * 8
|
component.heightStride = state.numMcuHigh * component.xScale * 8
|
||||||
component.channel =
|
component.channel = newMask(component.widthStride, component.heightStride)
|
||||||
newSeq[uint8](component.widthStride * component.heightStride)
|
|
||||||
|
|
||||||
if state.progressive:
|
if state.progressive:
|
||||||
component.widthCoeff = component.widthStride div 8
|
component.widthCoeff = component.widthStride div 8
|
||||||
|
@ -813,14 +849,14 @@ proc idctBlock(component: var Component, offset: int, data: array[64, int16]) =
|
||||||
x2 += 65536 + (128 shl 17)
|
x2 += 65536 + (128 shl 17)
|
||||||
x3 += 65536 + (128 shl 17)
|
x3 += 65536 + (128 shl 17)
|
||||||
|
|
||||||
component.channel[outPos + 0] = clampByte((x0 + t3) shr 17)
|
component.channel.data[outPos + 0] = clampByte((x0 + t3) shr 17)
|
||||||
component.channel[outPos + 7] = clampByte((x0 - t3) shr 17)
|
component.channel.data[outPos + 7] = clampByte((x0 - t3) shr 17)
|
||||||
component.channel[outPos + 1] = clampByte((x1 + t2) shr 17)
|
component.channel.data[outPos + 1] = clampByte((x1 + t2) shr 17)
|
||||||
component.channel[outPos + 6] = clampByte((x1 - t2) shr 17)
|
component.channel.data[outPos + 6] = clampByte((x1 - t2) shr 17)
|
||||||
component.channel[outPos + 2] = clampByte((x2 + t1) shr 17)
|
component.channel.data[outPos + 2] = clampByte((x2 + t1) shr 17)
|
||||||
component.channel[outPos + 5] = clampByte((x2 - t1) shr 17)
|
component.channel.data[outPos + 5] = clampByte((x2 - t1) shr 17)
|
||||||
component.channel[outPos + 3] = clampByte((x3 + t0) shr 17)
|
component.channel.data[outPos + 3] = clampByte((x3 + t0) shr 17)
|
||||||
component.channel[outPos + 4] = clampByte((x3 - t0) shr 17)
|
component.channel.data[outPos + 4] = clampByte((x3 - t0) shr 17)
|
||||||
|
|
||||||
{.pop.}
|
{.pop.}
|
||||||
|
|
||||||
|
@ -902,59 +938,45 @@ proc quantizationAndIDCTPass(state: var DecoderState) =
|
||||||
data
|
data
|
||||||
)
|
)
|
||||||
|
|
||||||
template dataIndex(component: var Component, x, y: int): int =
|
proc magnifyXBy2(mask: Mask): Mask =
|
||||||
component.widthStride * y + x
|
|
||||||
|
|
||||||
template `[]`*(component: Component, x, y: int): uint8 =
|
|
||||||
component.channel[component.dataIndex(x, y)]
|
|
||||||
|
|
||||||
proc magnifyXBy2(component: var Component) =
|
|
||||||
## Smooth magnify by power of 2 only in the X direction.
|
## Smooth magnify by power of 2 only in the X direction.
|
||||||
var magnified =
|
result = newMask(mask.width * 2, mask.height)
|
||||||
newSeq[uint8](component.widthStride * 2 * component.heightStride)
|
for y in 0 ..< mask.height:
|
||||||
for y in 0 ..< component.heightStride:
|
for x in 0 ..< mask.width:
|
||||||
for x in 0 ..< component.widthStride:
|
let n = 3 * mask.unsafe[x, y].uint16
|
||||||
let n = 3 * component[x, y].uint16
|
|
||||||
if x == 0:
|
if x == 0:
|
||||||
magnified[component.dataIndex(x * 2 + 0, y)] = component[x, y]
|
result.unsafe[x * 2 + 0, y] = mask.unsafe[x, y]
|
||||||
magnified[component.dataIndex(x * 2 + 1, y)] =
|
result.unsafe[x * 2 + 1, y] =
|
||||||
((n + component[x + 1, y].uint16 + 2) div 4).uint8
|
((n + mask.unsafe[x + 1, y].uint16 + 2) div 4).uint8
|
||||||
elif x == component.widthStride - 1:
|
elif x == mask.width - 1:
|
||||||
magnified[component.dataIndex(x * 2 + 0, y)] =
|
result.unsafe[x * 2 + 0, y] =
|
||||||
((n + component[x - 1, y].uint16 + 2) div 4).uint8
|
((n + mask.unsafe[x - 1, y].uint16 + 2) div 4).uint8
|
||||||
magnified[component.dataIndex(x * 2 + 1, y)] = component[x, y]
|
result.unsafe[x * 2 + 1, y] = mask.unsafe[x, y]
|
||||||
else:
|
else:
|
||||||
magnified[component.dataIndex(x * 2 + 0, y)] =
|
result.unsafe[x * 2 + 0, y] =
|
||||||
((n + component[x - 1, y].uint16) div 4).uint8
|
((n + mask.unsafe[x - 1, y].uint16) div 4).uint8
|
||||||
magnified[component.dataIndex(x * 2 + 1, y)] =
|
result.unsafe[x * 2 + 1, y] =
|
||||||
((n + component[x + 1, y].uint16) div 4).uint8
|
((n + mask.unsafe[x + 1, y].uint16) div 4).uint8
|
||||||
|
|
||||||
component.channel = move magnified
|
proc magnifyYBy2(mask: Mask): Mask =
|
||||||
component.widthStride *= 2
|
|
||||||
|
|
||||||
proc magnifyYBy2(component: var Component) =
|
|
||||||
## Smooth magnify by power of 2 only in the Y direction.
|
## Smooth magnify by power of 2 only in the Y direction.
|
||||||
var magnified =
|
result = newMask(mask.width, mask.height * 2)
|
||||||
newSeq[uint8](component.widthStride * component.heightStride * 2)
|
for y in 0 ..< mask.height:
|
||||||
for y in 0 ..< component.heightStride:
|
for x in 0 ..< mask.width:
|
||||||
for x in 0 ..< component.widthStride:
|
let n = 3 * mask.unsafe[x, y].uint16
|
||||||
let n = 3 * component[x, y].uint16
|
|
||||||
if y == 0:
|
if y == 0:
|
||||||
magnified[component.dataIndex(x, y * 2 + 0)] = component[x, y]
|
result.unsafe[x, y * 2 + 0] = mask.unsafe[x, y]
|
||||||
magnified[component.dataIndex(x, y * 2 + 1)] =
|
result.unsafe[x, y * 2 + 1] =
|
||||||
((n + component[x, y + 1].uint16 + 2) div 4).uint8
|
((n + mask.unsafe[x, y + 1].uint16 + 2) div 4).uint8
|
||||||
elif y == component.heightStride - 1:
|
elif y == mask.height - 1:
|
||||||
magnified[component.dataIndex(x, y * 2 + 0)] =
|
result.unsafe[x, y * 2 + 0] =
|
||||||
((n + component[x, y - 1].uint16 + 2) div 4).uint8
|
((n + mask.unsafe[x, y - 1].uint16 + 2) div 4).uint8
|
||||||
magnified[component.dataIndex(x, y * 2 + 1)] = component[x, y]
|
result.unsafe[x, y * 2 + 1] = mask.unsafe[x, y]
|
||||||
else:
|
else:
|
||||||
magnified[component.dataIndex(x, y * 2 + 0)] =
|
result.unsafe[x, y * 2 + 0] =
|
||||||
((n + component[x, y - 1].uint16) div 4).uint8
|
((n + mask.unsafe[x, y - 1].uint16) div 4).uint8
|
||||||
magnified[component.dataIndex(x, y * 2 + 1)] =
|
result.unsafe[x, y * 2 + 1] =
|
||||||
((n + component[x, y + 1].uint16) div 4).uint8
|
((n + mask.unsafe[x, y + 1].uint16) div 4).uint8
|
||||||
|
|
||||||
component.channel = move magnified
|
|
||||||
component.heightStride *= 2
|
|
||||||
|
|
||||||
proc yCbCrToRgbx(py, pcb, pcr: uint8): ColorRGBX =
|
proc yCbCrToRgbx(py, pcb, pcr: uint8): ColorRGBX =
|
||||||
## Takes a 3 component yCbCr outputs and populates image.
|
## Takes a 3 component yCbCr outputs and populates image.
|
||||||
|
@ -989,29 +1011,33 @@ proc buildImage(state: var DecoderState): Image =
|
||||||
of 3:
|
of 3:
|
||||||
for component in state.components.mitems:
|
for component in state.components.mitems:
|
||||||
while component.yScale < state.maxYScale:
|
while component.yScale < state.maxYScale:
|
||||||
component.magnifyXBy2()
|
component.channel = component.channel.magnifyXBy2()
|
||||||
component.yScale *= 2
|
component.yScale *= 2
|
||||||
|
|
||||||
while component.xScale < state.maxXScale:
|
while component.xScale < state.maxXScale:
|
||||||
component.magnifyYBy2()
|
component.channel = component.channel.magnifyYBy2()
|
||||||
component.xScale *= 2
|
component.xScale *= 2
|
||||||
|
|
||||||
|
let
|
||||||
|
cy = state.components[0].channel
|
||||||
|
cb = state.components[1].channel
|
||||||
|
cr = state.components[2].channel
|
||||||
for y in 0 ..< state.imageHeight:
|
for y in 0 ..< state.imageHeight:
|
||||||
var channelIndex = state.components[0].dataIndex(0, y)
|
var channelIndex = cy.dataIndex(0, y)
|
||||||
for x in 0 ..< state.imageWidth:
|
for x in 0 ..< state.imageWidth:
|
||||||
result.unsafe[x, y] = yCbCrToRgbx(
|
result.unsafe[x, y] = yCbCrToRgbx(
|
||||||
state.components[0].channel[channelIndex], # cy
|
cy.data[channelIndex],
|
||||||
state.components[1].channel[channelIndex], # cb
|
cb.data[channelIndex],
|
||||||
state.components[2].channel[channelIndex], # cr
|
cr.data[channelIndex],
|
||||||
)
|
)
|
||||||
inc channelIndex
|
inc channelIndex
|
||||||
|
|
||||||
of 1:
|
of 1:
|
||||||
|
let cy = state.components[0].channel
|
||||||
for y in 0 ..< state.imageHeight:
|
for y in 0 ..< state.imageHeight:
|
||||||
var channelIndex = state.components[0].dataIndex(0, y)
|
var channelIndex = cy.dataIndex(0, y)
|
||||||
for x in 0 ..< state.imageWidth:
|
for x in 0 ..< state.imageWidth:
|
||||||
result.unsafe[x, y] =
|
result.unsafe[x, y] = grayScaleToRgbx(cy.data[channelIndex])
|
||||||
grayScaleToRgbx(state.components[0].channel[channelIndex]) # cy
|
|
||||||
inc channelIndex
|
inc channelIndex
|
||||||
|
|
||||||
else:
|
else:
|
||||||
|
|
Loading…
Reference in a new issue