-
-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathgid-headers.adb
812 lines (769 loc) · 26.8 KB
/
gid-headers.adb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
---------------------------------
-- GID - Generic Image Decoder --
---------------------------------
--
-- Private child of GID, with helpers for identifying
-- image formats and reading header informations.
--
with GID.Buffering,
GID.Color_tables,
GID.Decoding_BMP,
GID.Decoding_JPG,
GID.Decoding_PNG,
GID.Decoding_PNM;
with Ada.Text_IO, Ada.Unchecked_Deallocation;
package body GID.Headers is
use Interfaces;
-------------------------------------------------------
-- The very first: read signature to identify format --
-------------------------------------------------------
procedure Load_Signature
(image : in out Image_Descriptor;
try_tga : Boolean := False)
is
use Bounded_255;
c : Character;
FITS_challenge : String (1 .. 5); -- without the initial
GIF_challenge : String (1 .. 5); -- without the initial
PNG_challenge : String (1 .. 7); -- without the initial
PNG_signature : constant String := "PNG" & ASCII.CR & ASCII.LF & ASCII.SUB & ASCII.LF;
PNM_challenge : Character;
QOI_challenge : String (1 .. 3); -- without the initial
QOI_signature : constant String := "oif";
RIFF_challenge : String (1 .. 3); -- without the initial
RIFF_signature : constant String := "IFF";
TIFF_challenge : String (1 .. 3); -- without the initial
TIFF_signature : String (1 .. 2);
procedure Dispose is
new Ada.Unchecked_Deallocation (Color_Table, p_Color_Table);
begin
-- Some cleanup
Dispose (image.palette);
image.next_frame := 0.0;
image.display_orientation := Unchanged;
--
Character'Read (image.stream, c);
image.first_byte := Character'Pos (c);
case c is
when 'B' =>
Character'Read (image.stream, c);
if c = 'M' then
image.detailed_format := To_Bounded_String ("BMP");
image.format := BMP;
return;
end if;
when 'S' =>
String'Read (image.stream, FITS_challenge);
if FITS_challenge = "IMPLE" then
image.detailed_format := To_Bounded_String ("FITS");
image.format := FITS;
return;
end if;
when 'G' =>
String'Read (image.stream, GIF_challenge);
if GIF_challenge = "IF87a" or GIF_challenge = "IF89a" then
image.detailed_format := To_Bounded_String ('G' & GIF_challenge & ", ");
image.format := GIF;
return;
end if;
when 'I' | 'M' =>
String'Read (image.stream, TIFF_challenge);
if c = TIFF_challenge (1) then
-- TIFF begins either with II (Intel) or MM (Motorola) - TIFF 6.0 Specification p.13
if c = 'I' then
image.detailed_format := To_Bounded_String ("TIFF, little-endian");
image.endianess := little;
TIFF_signature := '*' & ASCII.NUL; -- 42 (The Answer) on 16 bits
else
image.detailed_format := To_Bounded_String ("TIFF, big-endian");
image.endianess := big;
TIFF_signature := ASCII.NUL & '*'; -- 42 (The Answer) on 16 bits
end if;
if TIFF_challenge (2 .. 3) = TIFF_signature then
image.format := TIFF;
return;
end if;
end if;
when Character'Val (16#FF#) =>
Character'Read (image.stream, c);
if c = Character'Val (16#D8#) then
-- SOI (Start of Image) segment marker (FFD8)
image.detailed_format := To_Bounded_String ("JPEG");
image.format := JPEG;
return;
end if;
when Character'Val (16#89#) =>
String'Read (image.stream, PNG_challenge);
if PNG_challenge = PNG_signature then
image.detailed_format := To_Bounded_String ("PNG");
image.format := PNG;
return;
end if;
when 'P' =>
Character'Read (image.stream, PNM_challenge);
if PNM_challenge in '1' .. '6' then
image.detailed_format := To_Bounded_String ("PNM (PBM, PGM or PPM)");
image.format := PNM;
image.subformat_id := Integer'Value ((1 => PNM_challenge));
return;
end if;
when 'q' =>
String'Read (image.stream, QOI_challenge);
if QOI_challenge = QOI_signature then
image.format := QOI;
image.RLE_encoded := True;
return;
end if;
when 'R' =>
String'Read (image.stream, RIFF_challenge);
if RIFF_challenge = RIFF_signature then
image.format := RIFF;
return;
end if;
when others =>
if try_tga then
image.detailed_format := To_Bounded_String ("TGA");
image.format := TGA;
return;
else
raise unknown_image_format;
end if;
end case;
raise unknown_image_format;
end Load_Signature;
-- Define reading of unsigned numbers from a byte stream
-- Little-endian
generic
type Number_LE is mod <>;
procedure Read_Intel_x86_number (
from_le : in Stream_Access;
n : out Number_LE
);
pragma Inline (Read_Intel_x86_number);
generic
type Number_BE is mod <>;
procedure Big_endian_number (
from_be : in Stream_Access;
n : out Number_BE
);
pragma Inline (Big_endian_number);
generic
type Number is mod <>;
procedure Big_endian_number_buffered (
from : in out Input_Buffer;
n : out Number
);
pragma Inline (Big_endian_number_buffered);
generic
type Number is mod <>;
procedure Read_any_endian_number (
from : in Stream_Access;
n : out Number;
endi : in Endianess_Type
);
pragma Inline (Read_any_endian_number);
-- Implementations
procedure Read_Intel_x86_number (
from_le : in Stream_Access;
n : out Number_LE
)
is
b : U8;
m : Number_LE := 1;
begin
n := 0;
for i in 1 .. Number_LE'Size / 8 loop
U8'Read (from_le, b);
n := n + m * Number_LE (b);
m := m * 256;
end loop;
end Read_Intel_x86_number;
procedure Big_endian_number (
from_be : in Stream_Access;
n : out Number_BE
)
is
b : U8;
begin
n := 0;
for i in 1 .. Number_BE'Size / 8 loop
U8'Read (from_be, b);
n := n * 256 + Number_BE (b);
end loop;
end Big_endian_number;
procedure Big_endian_number_buffered (
from : in out Input_Buffer;
n : out Number
)
is
b : U8;
begin
n := 0;
for i in 1 .. Number'Size / 8 loop
Buffering.Get_Byte (from, b);
n := n * 256 + Number (b);
end loop;
end Big_endian_number_buffered;
procedure Read_any_endian_number (
from : in Stream_Access;
n : out Number;
endi : in Endianess_Type
)
is
procedure Read_Intel is new Read_Intel_x86_number (Number);
procedure Big_endian is new Big_endian_number (Number);
begin
case endi is
when little => Read_Intel (from, n);
when big => Big_endian (from, n);
end case;
end Read_any_endian_number;
-- Instantiations
procedure Read_Intel is new Read_Intel_x86_number (U16);
procedure Read_Intel is new Read_Intel_x86_number (U32);
procedure Big_endian_buffered is new Big_endian_number_buffered (U32);
procedure Read_any_endian is new Read_any_endian_number (U32);
----------------------------------------------------------
-- Loading of various format's headers (past signature) --
----------------------------------------------------------
----------------
-- BMP header --
----------------
procedure Load_BMP_Header (image : in out Image_Descriptor) is
n, dummy, header_size : U32;
w, dummy16 : U16;
dummy8 : U8;
use Bounded_255, Decoding_BMP;
compression : BMP_Compression;
begin
-- Pos= 3, read the file size
Read_Intel (image.stream, dummy);
-- Pos= 7, read four bytes, unknown
Read_Intel (image.stream, dummy);
-- Pos= 11, read four bytes offset, file top to bitmap data.
-- For 256 colors, this is usually 36 04 00 00
Read_Intel (image.stream, dummy);
-- Pos= 15. The beginning of Bitmap information header.
-- BITMAPINFOHEADER, BITMAPV5HEADER
-- biSize, bV5Size
Read_Intel (image.stream, header_size);
case header_size is
when 0 .. 39 =>
raise error_in_image_data
with "BMP Bitmap Info Header is too small:" & header_size'Image;
when 40 .. 51 => null;
when 52 .. 107 => Append (image.detailed_format, " v2");
when 108 .. 123 => Append (image.detailed_format, " v4");
when 124 .. U32'Last => Append (image.detailed_format, " v5");
end case;
-- Pos= 19. Bitmap width, in pixels: biWidth, bV5Width
Read_Intel (image.stream, n);
image.width := Positive_32 (n);
-- Pos= 23. Bitmap height, in pixels: biHeight, bV5Height
Read_Intel (image.stream, n);
image.height := Positive_32 (n);
-- Pos= 27: Bitmap planes: biPlanes, bV5Planes
Read_Intel (image.stream, w);
-- "The number of planes for the target device.
-- This value must be set to 1."
if w /= 1 then
raise error_in_image_data
with "BMP: invalid number of planes:" & w'Image & "; must be 1";
end if;
-- Pos= 29, Number of bits per pixel: biBitCount, bV5BitCount
Read_Intel (image.stream, w);
case w is
when 1 | 4 | 8 | 24 =>
null;
when 0 | 16 | 32 =>
raise unsupported_image_subformat with "BMP bit depth" & w'Image;
when others =>
raise error_in_image_data with "BMP invalid bit depth" & w'Image;
end case;
image.bits_per_pixel := Integer (w);
-- Pos= 31, Type of compression: biCompression, bV5Compression
Read_Intel (image.stream, n);
if n > BMP_Compression'Pos (BMP_Compression'Last) then
raise unsupported_image_subformat
with "BMP: unknown compression code" & n'Image;
else
compression := BMP_Compression'Val (n);
case compression is
when BI_RGB =>
image.RLE_encoded := False;
when BI_RLE8 =>
image.RLE_encoded := True;
if image.bits_per_pixel /= 8 then
raise error_in_image_data
with
"BMP: for BI_RLE8 compression, invalid number of bpp:" &
image.bits_per_pixel'Image & "; must be 8";
end if;
when BI_RLE4 =>
if image.bits_per_pixel /= 4 then
raise error_in_image_data
with
"BMP: for BI_RLE4 compression, invalid number of bpp:" &
image.bits_per_pixel'Image & "; must be 4";
end if;
when others =>
image.RLE_encoded := False;
end case;
case compression is
when BI_RGB =>
null; -- Supported - !! TBD: support BI_RLE8, BI_RLE4.
when others =>
raise unsupported_image_subformat
with "BMP: unsupported compression: " & compression'Image;
end case;
end if;
--
-- Pos= 35, Image size: biSizeImage, bV5SizeImage
Read_Intel (image.stream, dummy);
-- Pos= 39, horizontal resolution: biXPelsPerMeter, bV5XPelsPerMeter
Read_Intel (image.stream, dummy);
-- Pos= 43, vertical resolution: biYPelsPerMeter, bV5YPelsPerMeter
Read_Intel (image.stream, dummy);
-- Pos= 47, number of palette colors: biClrUsed, bV5ClrUsed
Read_Intel (image.stream, n);
if image.bits_per_pixel <= 8 then
if n = 0 then
image.palette := new Color_Table (0 .. 2**image.bits_per_pixel - 1);
else
image.palette := new Color_Table (0 .. Natural (n) - 1);
end if;
end if;
-- Pos= 51, number of important colors: biClrImportant, bV5ClrImportant
Read_Intel (image.stream, dummy);
-- Skip the rest of the header.
for skip in 41 .. header_size loop
U8'Read (image.stream, dummy8);
end loop;
-- Start of palette
Color_tables.Load_palette (image);
end Load_BMP_Header;
-----------------
-- FITS header --
-----------------
procedure Load_FITS_Header (image : in out Image_Descriptor) is
begin
raise known_but_unsupported_image_format
with "FITS format not yet supported";
end Load_FITS_Header;
----------------
-- GIF header --
----------------
procedure Load_GIF_Header (image : in out Image_Descriptor) is
-- GIF - logical screen descriptor
screen_width, screen_height : U16;
packed, background, aspect_ratio_code : U8;
global_palette : Boolean;
begin
Read_Intel (image.stream, screen_width);
Read_Intel (image.stream, screen_height);
if screen_width = 0 then
raise error_in_image_data with "GIF image: zero width";
end if;
if screen_height = 0 then
raise error_in_image_data with "GIF image: zero height";
end if;
image.width := Positive_32 (screen_width);
image.height := Positive_32 (screen_height);
image.transparency := True; -- cannot exclude transparency at this level.
U8'Read (image.stream, packed);
-- Global Color Table Flag 1 Bit
-- Color Resolution 3 Bits
-- Sort Flag 1 Bit
-- Size of Global Color Table 3 Bits
global_palette := (packed and 16#80#) /= 0;
image.bits_per_pixel := Natural ((packed and 16#7F#) / 16#10#) + 1;
-- Indicative:
-- iv) [...] This value should be set to indicate the
-- richness of the original palette
U8'Read (image.stream, background);
U8'Read (image.stream, aspect_ratio_code);
Buffering.Attach_Stream (image.buffer, image.stream);
if global_palette then
image.subformat_id := 1 + (Natural (packed and 16#07#));
-- palette's bits per pixels, usually <= image's
--
-- if image.subformat_id > image.bits_per_pixel then
-- raise
-- error_in_image_data with
-- "GIF: global palette has more colors than the image" &
-- image.subformat_id'img & image.bits_per_pixel'img;
-- end if;
image.palette := new Color_Table (0 .. 2**(image.subformat_id) - 1);
Color_tables.Load_palette (image);
end if;
end Load_GIF_Header;
-----------------
-- JPEG header --
-----------------
procedure Load_JPEG_Header (image : in out Image_Descriptor) is
use Decoding_JPG, Buffering, Ada.Text_IO;
head : Segment_Head;
b : U8;
begin
-- We have already passed the SOI (Start of Image) segment marker (FFD8).
image.JPEG_stuff.restart_interval := 0;
Attach_Stream (image.buffer, image.stream);
loop
Read (image, False, 0, head);
case head.kind is
when DHT =>
-- Huffman Table
Read_DHT (image, Natural (head.length));
when DQT =>
Read_DQT (image, Natural (head.length));
when DRI =>
-- Restart Interval
Read_DRI (image);
when SOF_0 .. SOF_15 =>
Read_SOF (image, head);
-- We've got frame-header-style informations (SOF),
-- then it's time to quit:
exit;
when APP_1 =>
Read_EXIF (image, Natural (head.length));
when COM =>
-- B.2.4.5 Comment
if some_trace then
New_Line;
Put_Line ("JPEG Comment (during Load_JPEG_Header): --------");
for i in 1 .. head.length loop
Get_Byte (image.buffer, b);
Put (Character'Val (b));
end loop;
New_Line;
Put_Line ("-------------------------------------------------");
New_Line;
else
Skip_Segment_Data (image, head);
end if;
when others =>
Skip_Segment_Data (image, head);
end case;
end loop;
end Load_JPEG_Header;
----------------
-- PNG header --
----------------
procedure Load_PNG_Header (image : in out Image_Descriptor) is
use Decoding_PNG, Buffering;
ch : Chunk_Header;
n, dummy : U32;
b, color_type : U8;
procedure Read_Palette is
begin
loop
Read_Chunk_Header (image, ch);
case ch.kind is
when IEND =>
raise error_in_image_data with
"PNG: a palette (PLTE) is expected here, found IEND";
when PLTE =>
if ch.length rem 3 /= 0 then
raise error_in_image_data with
"PNG: palette chunk byte length must be a multiple of 3";
end if;
image.palette := new Color_Table (0 .. Integer (ch.length / 3) - 1);
Color_tables.Load_palette (image);
Big_endian_buffered (image.buffer, dummy); -- Chunk's CRC
exit;
when others =>
-- Skip chunk data and CRC
for i in 1 .. ch.length + 4 loop
Get_Byte (image.buffer, b);
end loop;
end case;
end loop;
end Read_Palette;
palette : Boolean := False;
begin
Buffering.Attach_Stream (image.buffer, image.stream);
Read_Chunk_Header (image, ch);
if ch.kind /= IHDR then
raise error_in_image_data with "PNG: expected 'IHDR' chunk as first chunk in PNG stream";
end if;
Big_endian_buffered (image.buffer, n);
if n = 0 then
raise error_in_image_data with "PNG image with zero width";
end if;
if n > U32 (Positive_32'Last) then
raise error_in_image_data with "PNG image: width value too large:" & U32'Image (n);
end if;
image.width := Positive_32 (n);
Big_endian_buffered (image.buffer, n);
if n = 0 then
raise error_in_image_data with "PNG image with zero height";
end if;
if n > U32 (Positive_32'Last) then
raise error_in_image_data with "PNG image: height value too large:" & U32'Image (n);
end if;
image.height := Positive_32 (n);
Get_Byte (image.buffer, b);
if b = 0 then
raise error_in_image_data with "PNG image: zero bit-per-pixel";
end if;
image.bits_per_pixel := Integer (b);
Get_Byte (image.buffer, color_type);
image.subformat_id := Integer (color_type);
case color_type is
when 0 => -- Greyscale
image.greyscale := True;
case image.bits_per_pixel is
when 1 | 2 | 4 | 8 | 16 =>
null;
when others =>
raise error_in_image_data with
"PNG, type 0 (greyscale): wrong bit-per-channel depth";
end case;
when 2 => -- RGB TrueColor
case image.bits_per_pixel is
when 8 | 16 =>
image.bits_per_pixel := 3 * image.bits_per_pixel;
when others =>
raise error_in_image_data with
"PNG, type 2 (RGB): wrong bit-per-channel depth";
end case;
when 3 => -- RGB with palette
palette := True;
case image.bits_per_pixel is
when 1 | 2 | 4 | 8 =>
null;
when others =>
raise error_in_image_data with
"PNG, type 3: wrong bit-per-channel depth";
end case;
when 4 => -- Grey & Alpha
image.greyscale := True;
image.transparency := True;
case image.bits_per_pixel is
when 8 | 16 =>
image.bits_per_pixel := 2 * image.bits_per_pixel;
when others =>
raise error_in_image_data with
"PNG, type 4 (Greyscale & Alpha): wrong bit-per-channel depth";
end case;
when 6 => -- RGBA
image.transparency := True;
case image.bits_per_pixel is
when 8 | 16 =>
image.bits_per_pixel := 4 * image.bits_per_pixel;
when others =>
raise error_in_image_data with
"PNG, type 6 (RGBA): wrong bit-per-channel depth";
end case;
when others =>
raise error_in_image_data with "PNG: unknown color type";
end case;
Get_Byte (image.buffer, b);
if b /= 0 then
raise error_in_image_data with
"PNG: unknown compression format; ISO/IEC 15948:2003" &
" knows only 'method 0' (Deflate)";
end if;
Get_Byte (image.buffer, b);
if b /= 0 then
raise error_in_image_data with
"PNG: unknown filtering; ISO/IEC 15948:2003 knows only 'method 0'";
end if;
Get_Byte (image.buffer, b);
image.progressive := b = 1; -- Adam7
Big_endian_buffered (image.buffer, dummy); -- Chunk's CRC
if palette then
Read_Palette;
end if;
image.PNG_stuff :=
(frame_width => image.width,
frame_height => image.height,
x_offset => 0,
y_offset => 0,
dispose_op => PNG_Defs.APNG_DISPOSE_OP_NONE,
blend_op => PNG_Defs.APNG_BLEND_OP_SOURCE);
end Load_PNG_Header;
--------------------------------
-- PNM (PBM, PGM, PPM) header --
--------------------------------
procedure Load_PNM_Header (image : in out Image_Descriptor) is
use Decoding_PNM;
depth_val : Integer;
begin
image.width := Get_Positive_32 (image.stream);
case image.subformat_id is
when 1 | 4 =>
image.height := Get_Positive_32 (image.stream, needs_EOL => True);
image.greyscale := True;
image.bits_per_pixel := 3;
when 2 .. 3 | 5 .. 6 =>
image.height := Get_Positive_32 (image.stream);
depth_val := Get_Integer (image.stream, needs_EOL => True);
if depth_val /= 255 then
raise unsupported_image_subformat with
"PNM: maximum depth value" & Integer'Image (depth_val) &
"; only 255 is supported";
end if;
image.greyscale := image.subformat_id = 2 or image.subformat_id = 5;
image.bits_per_pixel := 24;
when others =>
raise unsupported_image_subformat with
"PNM: P" & Integer'Image (image.subformat_id) & " not supported";
end case;
exception
when Constraint_Error =>
raise error_in_image_data with "PNM: invalid numeric value in PNM header";
end Load_PNM_Header;
----------------
-- QOI header --
----------------
procedure Load_QOI_Header (image : in out Image_Descriptor) is
val_32 : U32;
channels, colorspace : U8;
begin
Buffering.Attach_Stream (image.buffer, image.stream);
Read_any_endian (image.stream, val_32, big);
image.width := Positive_32 (val_32);
Read_any_endian (image.stream, val_32, big);
image.height := Positive_32 (val_32);
U8'Read (image.stream, channels);
image.bits_per_pixel := Positive (channels) * 8;
image.transparency := channels = 4;
U8'Read (image.stream, colorspace);
end Load_QOI_Header;
-----------------
-- RIFF header --
-----------------
procedure Load_RIFF_Header (image : in out Image_Descriptor) is
begin
raise known_but_unsupported_image_format
with "RIFF (for WebP) not yet supported";
end Load_RIFF_Header;
------------------------
-- TGA (Targa) header --
------------------------
procedure Load_TGA_Header (image : in out Image_Descriptor) is
-- TGA FILE HEADER, p.6
--
image_ID_length : U8; -- Field 1
color_map_type : U8; -- Field 2
image_type : U8; -- Field 3
-- Color Map Specification - Field 4
first_entry_index : U16; -- Field 4.1
color_map_length : U16; -- Field 4.2
color_map_entry_size : U8; -- Field 4.3
-- Image Specification - Field 5
x_origin : U16;
y_origin : U16;
image_width : U16;
image_height : U16;
pixel_depth : U8;
tga_image_descriptor : U8;
--
dummy : U8;
base_image_type : Integer;
begin
-- Read the header
image_ID_length := image.first_byte;
U8'Read (image.stream, color_map_type);
U8'Read (image.stream, image_type);
-- Color Map Specification - Field 4
Read_Intel (image.stream, first_entry_index);
Read_Intel (image.stream, color_map_length);
U8'Read (image.stream, color_map_entry_size);
-- Image Specification - Field 5
Read_Intel (image.stream, x_origin);
Read_Intel (image.stream, y_origin);
Read_Intel (image.stream, image_width);
Read_Intel (image.stream, image_height);
U8'Read (image.stream, pixel_depth);
U8'Read (image.stream, tga_image_descriptor);
-- Done.
--
-- Image type:
-- 1 = 8-bit palette style
-- 2 = Direct [A]RGB image
-- 3 = grayscale
-- 9 = RLE version of Type 1
-- 10 = RLE version of Type 2
-- 11 = RLE version of Type 3
--
base_image_type := U8'Pos (image_type and 7);
image.RLE_encoded := (image_type and 8) /= 0;
--
if color_map_type /= 0 then
image.palette := new Color_Table
(Integer (first_entry_index) ..
Integer (first_entry_index) + Integer (color_map_length) - 1);
--
image.subformat_id := Integer (color_map_entry_size);
case image.subformat_id is -- = palette's bit depth
when 8 => -- Grey
null;
when 15 => -- RGB 3*5 bit
null;
when 16 => -- RGBA 3*5+1 bit
image.transparency := True;
when 24 => -- RGB 3*8 bit
null;
when 32 => -- RGBA 4*8 bit
image.transparency := True;
when others =>
raise error_in_image_data with
"TGA color map (palette): wrong bit depth:" &
Integer'Image (image.subformat_id);
end case;
end if;
--
image.greyscale := False; -- ev. overridden later
case base_image_type is
when 1 =>
image.greyscale := color_map_entry_size = 8;
when 2 =>
null;
when 3 =>
image.greyscale := True;
when others =>
raise unsupported_image_subformat with
"TGA type =" & Integer'Image (base_image_type);
end case;
image.width := U16'Pos (image_width);
image.height := U16'Pos (image_height);
image.bits_per_pixel := U8'Pos (pixel_depth);
-- Make sure we are loading a supported TGA_type
case image.bits_per_pixel is
when 24 | 15 | 8 =>
null;
when 32 | 16 =>
image.transparency := True;
when others =>
raise unsupported_image_subformat with
"TGA bits per pixels =" & Integer'Image (image.bits_per_pixel) &
"; supported bpp are: 8, 15, 16, 24, 32";
end case;
image.top_first := (tga_image_descriptor and 32) /= 0;
-- *** Image and color map data
-- * Image ID
for i in 1 .. image_ID_length loop
U8'Read (image.stream, dummy);
end loop;
-- * Color map data (palette)
Color_tables.Load_palette (image);
-- * Image data: Read by Load_image_contents.
end Load_TGA_Header;
procedure Load_TIFF_Header (image : in out Image_Descriptor) is
first_IFD_offset : U32;
--
-- IFD: Image File Directory. Basically, the image header.
-- Issue with TIFF: often the image header is stored after the image data.
-- This would need streams with Set_Index instead of a general stream
-- (e.g. a file, not an HTTP stream), or to store the full image data
-- in a temp buffer. Perhaps we'll do that one day.
begin
Read_any_endian (image.stream, first_IFD_offset, image.endianess);
raise known_but_unsupported_image_format with
"TIFF is not appropriate for streaming. " &
"Use PNG, BMP (lossless) or JPEG instead." &
"Info: IFD Offset=" & U32'Image (first_IFD_offset);
end Load_TIFF_Header;
end GID.Headers;