1 ! Copyright (C) 2009 Marc Fauconneau.
2 ! See http://factorcode.org/license.txt for BSD license.
3 USING: accessors arrays byte-arrays combinators
4 compression.huffman fry grouping images images.loader
5 images.processing io io.binary io.encodings.binary
6 io.streams.byte-array io.streams.limited io.streams.throwing
7 kernel locals math math.bitwise math.blas.matrices
8 math.blas.vectors math.constants math.functions math.matrices
9 math.order math.vectors memoize namespaces sequences
11 QUALIFIED-WITH: bitstreams bs
16 TUPLE: loading-jpeg < image
19 { color-info initial: { f f f f } }
20 { quant-tables initial: { f f } }
21 { huff-tables initial: { f f f f } }
24 "jpg" jpeg-image register-image-class
25 "jpeg" jpeg-image register-image-class
29 : <loading-jpeg> ( headers bitstream -- image )
30 loading-jpeg new swap >>bitstream swap >>headers ;
32 SINGLETONS: SOF DHT DAC RST SOI EOI SOS DQT DNL DRI DHP EXP
35 ! ISO/IEC 10918-1 Table B.1
36 :: >marker ( byte -- marker )
39 { [ dup 0xCC = ] [ { DAC } ] }
40 { [ dup 0xC4 = ] [ { DHT } ] }
41 { [ dup 0xC9 = ] [ { JPG } ] }
42 { [ dup -4 shift 0xC = ] [ SOF byte 4 bits 2array ] }
44 { [ dup 0xD8 = ] [ { SOI } ] }
45 { [ dup 0xD9 = ] [ { EOI } ] }
46 { [ dup 0xDA = ] [ { SOS } ] }
47 { [ dup 0xDB = ] [ { DQT } ] }
48 { [ dup 0xDC = ] [ { DNL } ] }
49 { [ dup 0xDD = ] [ { DRI } ] }
50 { [ dup 0xDE = ] [ { DHP } ] }
51 { [ dup 0xDF = ] [ { EXP } ] }
52 { [ dup -4 shift 0xD = ] [ RST byte 4 bits 2array ] }
54 { [ dup -4 shift 0xE = ] [ APP byte 4 bits 2array ] }
55 { [ dup 0xFE = ] [ { COM } ] }
56 { [ dup -4 shift 0xF = ] [ JPG byte 4 bits 2array ] }
58 { [ dup 0x01 = ] [ { TEM } ] }
63 TUPLE: jpeg-chunk length type data ;
65 : <jpeg-chunk> ( type length data -- jpeg-chunk )
71 TUPLE: jpeg-color-info
72 h v quant-table dc-huff-table ac-huff-table { diff initial: 0 } id ;
74 : <jpeg-color-info> ( h v quant-table -- jpeg-color-info )
80 : jpeg> ( -- jpeg-image ) jpeg-image get ;
82 : apply-diff ( dc color -- dc' )
83 [ diff>> + dup ] [ diff<< ] bi ;
85 : fetch-tables ( component -- )
86 [ [ jpeg> quant-tables>> nth ] change-quant-table drop ]
87 [ [ jpeg> huff-tables>> nth ] change-dc-huff-table drop ]
88 [ [ 2 + jpeg> huff-tables>> nth ] change-ac-huff-table drop ] tri ;
90 : read4/4 ( -- a b ) read1 16 /mod ;
94 : decode-frame ( header -- )
101 swap 2array jpeg> dim<<
104 read1 read4/4 read1 <jpeg-color-info>
105 swap [ >>id ] keep jpeg> color-info>> set-nth
109 : decode-quant-table ( chunk -- )
116 read4/4 [ 0 assert= ] dip
118 swap jpeg> quant-tables>> set-nth
122 : decode-huff-table ( chunk -- )
123 data>> [ binary <byte-reader> ] [ length ] bi limit-stream [
124 [ input-stream get stream>> [ count>> ] [ limit>> ] bi < ]
128 dup [ ] [ + ] map-reduce read
129 binary [ [ read [ B{ } ] unless* ] { } map-as ] with-byte-reader
130 swap jpeg> huff-tables>> set-nth
132 ] stream-throw-on-eof ;
134 : decode-scan ( chunk -- )
140 read1 jpeg> color-info>> nth clone
141 read1 16 /mod [ >>dc-huff-table ] [ >>ac-huff-table ] bi*
142 ] map jpeg> components<<
145 read1 16 /mod [ 0 assert= ] bi@
148 : singleton-first ( seq -- elt )
149 [ length 1 assert= ] [ first ] bi ;
151 ERROR: not-a-baseline-jpeg-image ;
153 : baseline-parse ( -- )
154 jpeg> headers>> [ type>> { SOF 0 } = ] any? [ not-a-baseline-jpeg-image ] unless
157 [ [ type>> { SOF 0 } = ] filter singleton-first decode-frame ]
158 [ [ type>> { DQT } = ] filter [ decode-quant-table ] each ]
159 [ [ type>> { DHT } = ] filter [ decode-huff-table ] each ]
160 [ [ type>> { SOS } = ] filter singleton-first decode-scan ]
163 : parse-marker ( -- marker )
167 : parse-headers ( -- chunks )
168 [ parse-marker dup { SOS } = not ]
171 dup 2 - read <jpeg-chunk>
172 ] [ produce ] keep dip swap suffix ;
174 MEMO: zig-zag ( -- zz )
176 { 0 1 5 6 14 15 27 28 }
177 { 2 4 7 13 16 26 29 42 }
178 { 3 8 12 17 25 30 41 43 }
179 { 9 11 18 24 31 40 44 53 }
180 { 10 19 23 32 39 45 52 54 }
181 { 20 22 33 38 46 51 55 60 }
182 { 21 34 37 47 50 56 59 61 }
183 { 35 36 48 49 57 58 62 63 }
186 MEMO: yuv>bgr-matrix ( -- m )
189 { 1 -0.39465 -0.58060 }
193 : wave ( x u -- n ) swap 2 * 1 + * pi * 16 / cos ;
195 :: dct-vect ( u v -- basis )
196 { 8 8 } coord-matrix [ { u v } [ wave ] 2map product ] map^2
197 1 u v [ 0 = [ 2 sqrt / ] when ] bi@ 4 / m*n ;
199 MEMO: dct-matrix ( -- m ) 64 iota [ 8 /mod dct-vect flatten ] map ;
201 : mb-dim ( component -- dim ) [ h>> ] [ v>> ] bi 2array ;
203 ! : blocks ( component -- seq )
204 ! mb-dim ! coord-matrix flip concat [ [ { 2 2 } v* ] [ v+ ] bi* ] with map ;
206 : all-macroblocks ( quot: ( mb -- ) -- )
210 [ color-info>> sift { 0 0 } [ mb-dim vmax ] reduce v/ ] bi
212 coord-matrix flip concat
214 [ each ] bi* ; inline
216 : reverse-zigzag ( b -- b' ) zig-zag swap [ nth ] curry map ;
218 : idct-factor ( b -- b' ) dct-matrix v.m ;
220 MEMO: dct-matrix-blas ( -- m ) dct-matrix >float-blas-matrix ;
221 : V.M ( x A -- x.A ) Mtranspose swap M.V ;
222 : idct-blas ( b -- b' ) >float-blas-vector dct-matrix-blas V.M ;
224 : idct ( b -- b' ) idct-factor ;
226 :: draw-block ( block x,y color-id jpeg-image -- )
227 block dup length>> sqrt >fixnum group flip
228 dup matrix-dim coord-matrix flip
230 [ '[ _ [ second ] [ first ] bi ] dip nth nth ]
231 [ x,y v+ color-id jpeg-image draw-color ] bi
234 : sign-extend ( bits v -- v' )
235 swap [ ] [ 1 - 2^ < ] 2bi
236 [ -1 swap shift 1 + + ] [ drop ] if ;
238 : read1-jpeg-dc ( decoder -- dc )
239 [ read1-huff dup ] [ bs>> bs:read ] bi sign-extend ;
241 : read1-jpeg-ac ( decoder -- run/ac )
242 [ read1-huff 16 /mod dup ] [ bs>> bs:read ] bi sign-extend 2array ;
244 :: decode-block ( color -- pixels )
245 color dc-huff-table>> read1-jpeg-dc color apply-diff
246 64 0 <array> :> coefs
250 color ac-huff-table>> read1-jpeg-ac
251 [ first 1 + k + k! ] [ second k coefs set-nth ] [ ] tri
255 coefs color quant-table>> v*
256 reverse-zigzag idct ;
258 :: draw-macroblock-yuv420 ( mb blocks -- )
259 mb { 16 16 } v* :> pos
260 0 blocks nth pos { 0 0 } v+ 0 jpeg> draw-block
261 1 blocks nth pos { 8 0 } v+ 0 jpeg> draw-block
262 2 blocks nth pos { 0 8 } v+ 0 jpeg> draw-block
263 3 blocks nth pos { 8 8 } v+ 0 jpeg> draw-block
264 4 blocks nth 8 group 2 matrix-zoom concat pos 1 jpeg> draw-block
265 5 blocks nth 8 group 2 matrix-zoom concat pos 2 jpeg> draw-block ;
267 :: draw-macroblock-yuv444 ( mb blocks -- )
269 3 iota [ [ blocks nth pos ] [ jpeg> draw-block ] bi ] each ;
271 :: draw-macroblock-y ( mb blocks -- )
273 0 blocks nth pos 0 jpeg> draw-block
274 64 0 <array> pos 1 jpeg> draw-block
275 64 0 <array> pos 2 jpeg> draw-block ;
279 ! [ 8 group 2 matrix-zoom concat ] unless
280 ! pos { 8 8 } v* color jpeg> draw-block ;
282 : decode-macroblock ( -- blocks )
286 [ [ decode-block ] curry replicate ] bi
289 : cleanup-bitstream ( bytes -- bytes' )
293 read1 [ 0x00 = and ] keep swap
296 swap >marker { EOI } assert=
301 : setup-bitmap ( image -- )
302 dup dim>> 16 v/n [ ceiling ] map 16 v*n >>dim
303 BGR >>component-order
304 ubyte-components >>component-type
306 dup dim>> first2 * 3 * 0 <array> >>bitmap
309 ERROR: unsupported-colorspace ;
310 SINGLETONS: YUV420 YUV444 Y MAGIC! ;
312 :: detect-colorspace ( jpeg-image -- csp )
313 jpeg-image color-info>> sift :> colors
315 colors length 1 = [ drop Y ] when
318 colors [ mb-dim { 1 1 } = ] all?
322 [ [ mb-dim { 1 1 } = ] all? ]
323 [ mb-dim { 2 2 } = ] bi* and
327 ! this eats ~50% cpu time
328 : draw-macroblocks ( mbs -- )
329 jpeg> detect-colorspace
331 { YUV420 [ [ first2 draw-macroblock-yuv420 ] each ] }
332 { YUV444 [ [ first2 draw-macroblock-yuv444 ] each ] }
333 { Y [ [ first2 draw-macroblock-y ] each ] }
334 [ unsupported-colorspace ]
337 ! this eats ~25% cpu time
338 : color-transform ( yuv -- rgb )
339 { 128 0 0 } v+ yuv>bgr-matrix swap m.v
340 [ 0 max 255 min >fixnum ] map ;
342 : baseline-decompress ( -- )
343 jpeg> bitstream>> cleanup-bitstream { 255 255 255 255 } append
344 >byte-array bs:<msb0-bit-reader> jpeg> bitstream<<
347 [ [ [ <huffman-decoder> ] with map ] change-huff-tables drop ] bi
348 jpeg> components>> [ fetch-tables ] each
349 [ decode-macroblock 2array ] collector
350 [ all-macroblocks ] dip
351 jpeg> setup-bitmap draw-macroblocks
352 jpeg> bitmap>> 3 <groups> [ color-transform ] map! drop
353 jpeg> [ >byte-array ] change-bitmap drop ;
355 ERROR: not-a-jpeg-image ;
357 : loading-jpeg>image ( loading-jpeg -- image )
363 : load-jpeg ( stream -- loading-jpeg )
365 parse-marker { SOI } = [ not-a-jpeg-image ] unless
367 contents <loading-jpeg>
368 ] with-input-stream ;
372 M: jpeg-image stream>image ( stream jpeg-image -- bitmap )
373 drop load-jpeg loading-jpeg>image ;