Coverage Report

Created: 2026-05-16 07:21

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/qtbase/src/gui/image/qxpmhandler.cpp
Line
Count
Source
1
// Copyright (C) 2016 The Qt Company Ltd.
2
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3
// Qt-Security score:critical reason:data-parser
4
5
#include "private/qxpmhandler_p.h"
6
7
#ifndef QT_NO_IMAGEFORMAT_XPM
8
9
#include <qbytearraymatcher.h>
10
#include <qdebug.h>
11
#include <qimage.h>
12
#include <qloggingcategory.h>
13
#include <qmap.h>
14
#include <qtextstream.h>
15
#include <qvariant.h>
16
17
#include <private/qcolor_p.h>
18
#include <private/qduplicatetracker_p.h> // for easier std::pmr detection
19
#include <private/qtools_p.h>
20
#include <private/qimage_p.h>
21
22
#include <algorithm>
23
#include <array>
24
25
QT_BEGIN_NAMESPACE
26
27
using namespace QtMiscUtils;
28
29
static quint64 xpmHash(const QString &str)
30
0
{
31
0
    unsigned int hashValue = 0;
32
0
    for (int i = 0; i < str.size(); ++i) {
33
0
        hashValue <<= 8;
34
0
        hashValue += (unsigned int)str.at(i).unicode();
35
0
    }
36
0
    return hashValue;
37
0
}
38
static quint64 xpmHash(char *str)
39
0
{
40
0
    unsigned int hashValue = 0;
41
0
    while (*str != '\0') {
42
0
        hashValue <<= 8;
43
0
        hashValue += (unsigned int)*str;
44
0
        ++str;
45
0
    }
46
0
    return hashValue;
47
0
}
48
49
#ifdef QRGB
50
#undef QRGB
51
#endif
52
#define QRGB(r,g,b) (r*65536 + g*256 + b)
53
54
static const int xpmRgbTblSize = 657;
55
56
static const struct XPMRGBData {
57
    uint  value;
58
    const char name[21];
59
} xpmRgbTbl[] = {
60
  { QRGB(240,248,255),  "aliceblue" },
61
  { QRGB(250,235,215),  "antiquewhite" },
62
  { QRGB(255,239,219),  "antiquewhite1" },
63
  { QRGB(238,223,204),  "antiquewhite2" },
64
  { QRGB(205,192,176),  "antiquewhite3" },
65
  { QRGB(139,131,120),  "antiquewhite4" },
66
  { QRGB(127,255,212),  "aquamarine" },
67
  { QRGB(127,255,212),  "aquamarine1" },
68
  { QRGB(118,238,198),  "aquamarine2" },
69
  { QRGB(102,205,170),  "aquamarine3" },
70
  { QRGB( 69,139,116),  "aquamarine4" },
71
  { QRGB(240,255,255),  "azure" },
72
  { QRGB(240,255,255),  "azure1" },
73
  { QRGB(224,238,238),  "azure2" },
74
  { QRGB(193,205,205),  "azure3" },
75
  { QRGB(131,139,139),  "azure4" },
76
  { QRGB(245,245,220),  "beige" },
77
  { QRGB(255,228,196),  "bisque" },
78
  { QRGB(255,228,196),  "bisque1" },
79
  { QRGB(238,213,183),  "bisque2" },
80
  { QRGB(205,183,158),  "bisque3" },
81
  { QRGB(139,125,107),  "bisque4" },
82
  { QRGB(  0,  0,  0),  "black" },
83
  { QRGB(255,235,205),  "blanchedalmond" },
84
  { QRGB(  0,  0,255),  "blue" },
85
  { QRGB(  0,  0,255),  "blue1" },
86
  { QRGB(  0,  0,238),  "blue2" },
87
  { QRGB(  0,  0,205),  "blue3" },
88
  { QRGB(  0,  0,139),  "blue4" },
89
  { QRGB(138, 43,226),  "blueviolet" },
90
  { QRGB(165, 42, 42),  "brown" },
91
  { QRGB(255, 64, 64),  "brown1" },
92
  { QRGB(238, 59, 59),  "brown2" },
93
  { QRGB(205, 51, 51),  "brown3" },
94
  { QRGB(139, 35, 35),  "brown4" },
95
  { QRGB(222,184,135),  "burlywood" },
96
  { QRGB(255,211,155),  "burlywood1" },
97
  { QRGB(238,197,145),  "burlywood2" },
98
  { QRGB(205,170,125),  "burlywood3" },
99
  { QRGB(139,115, 85),  "burlywood4" },
100
  { QRGB( 95,158,160),  "cadetblue" },
101
  { QRGB(152,245,255),  "cadetblue1" },
102
  { QRGB(142,229,238),  "cadetblue2" },
103
  { QRGB(122,197,205),  "cadetblue3" },
104
  { QRGB( 83,134,139),  "cadetblue4" },
105
  { QRGB(127,255,  0),  "chartreuse" },
106
  { QRGB(127,255,  0),  "chartreuse1" },
107
  { QRGB(118,238,  0),  "chartreuse2" },
108
  { QRGB(102,205,  0),  "chartreuse3" },
109
  { QRGB( 69,139,  0),  "chartreuse4" },
110
  { QRGB(210,105, 30),  "chocolate" },
111
  { QRGB(255,127, 36),  "chocolate1" },
112
  { QRGB(238,118, 33),  "chocolate2" },
113
  { QRGB(205,102, 29),  "chocolate3" },
114
  { QRGB(139, 69, 19),  "chocolate4" },
115
  { QRGB(255,127, 80),  "coral" },
116
  { QRGB(255,114, 86),  "coral1" },
117
  { QRGB(238,106, 80),  "coral2" },
118
  { QRGB(205, 91, 69),  "coral3" },
119
  { QRGB(139, 62, 47),  "coral4" },
120
  { QRGB(100,149,237),  "cornflowerblue" },
121
  { QRGB(255,248,220),  "cornsilk" },
122
  { QRGB(255,248,220),  "cornsilk1" },
123
  { QRGB(238,232,205),  "cornsilk2" },
124
  { QRGB(205,200,177),  "cornsilk3" },
125
  { QRGB(139,136,120),  "cornsilk4" },
126
  { QRGB(  0,255,255),  "cyan" },
127
  { QRGB(  0,255,255),  "cyan1" },
128
  { QRGB(  0,238,238),  "cyan2" },
129
  { QRGB(  0,205,205),  "cyan3" },
130
  { QRGB(  0,139,139),  "cyan4" },
131
  { QRGB(  0,  0,139),  "darkblue" },
132
  { QRGB(  0,139,139),  "darkcyan" },
133
  { QRGB(184,134, 11),  "darkgoldenrod" },
134
  { QRGB(255,185, 15),  "darkgoldenrod1" },
135
  { QRGB(238,173, 14),  "darkgoldenrod2" },
136
  { QRGB(205,149, 12),  "darkgoldenrod3" },
137
  { QRGB(139,101,  8),  "darkgoldenrod4" },
138
  { QRGB(169,169,169),  "darkgray" },
139
  { QRGB(  0,100,  0),  "darkgreen" },
140
  { QRGB(169,169,169),  "darkgrey" },
141
  { QRGB(189,183,107),  "darkkhaki" },
142
  { QRGB(139,  0,139),  "darkmagenta" },
143
  { QRGB( 85,107, 47),  "darkolivegreen" },
144
  { QRGB(202,255,112),  "darkolivegreen1" },
145
  { QRGB(188,238,104),  "darkolivegreen2" },
146
  { QRGB(162,205, 90),  "darkolivegreen3" },
147
  { QRGB(110,139, 61),  "darkolivegreen4" },
148
  { QRGB(255,140,  0),  "darkorange" },
149
  { QRGB(255,127,  0),  "darkorange1" },
150
  { QRGB(238,118,  0),  "darkorange2" },
151
  { QRGB(205,102,  0),  "darkorange3" },
152
  { QRGB(139, 69,  0),  "darkorange4" },
153
  { QRGB(153, 50,204),  "darkorchid" },
154
  { QRGB(191, 62,255),  "darkorchid1" },
155
  { QRGB(178, 58,238),  "darkorchid2" },
156
  { QRGB(154, 50,205),  "darkorchid3" },
157
  { QRGB(104, 34,139),  "darkorchid4" },
158
  { QRGB(139,  0,  0),  "darkred" },
159
  { QRGB(233,150,122),  "darksalmon" },
160
  { QRGB(143,188,143),  "darkseagreen" },
161
  { QRGB(193,255,193),  "darkseagreen1" },
162
  { QRGB(180,238,180),  "darkseagreen2" },
163
  { QRGB(155,205,155),  "darkseagreen3" },
164
  { QRGB(105,139,105),  "darkseagreen4" },
165
  { QRGB( 72, 61,139),  "darkslateblue" },
166
  { QRGB( 47, 79, 79),  "darkslategray" },
167
  { QRGB(151,255,255),  "darkslategray1" },
168
  { QRGB(141,238,238),  "darkslategray2" },
169
  { QRGB(121,205,205),  "darkslategray3" },
170
  { QRGB( 82,139,139),  "darkslategray4" },
171
  { QRGB( 47, 79, 79),  "darkslategrey" },
172
  { QRGB(  0,206,209),  "darkturquoise" },
173
  { QRGB(148,  0,211),  "darkviolet" },
174
  { QRGB(255, 20,147),  "deeppink" },
175
  { QRGB(255, 20,147),  "deeppink1" },
176
  { QRGB(238, 18,137),  "deeppink2" },
177
  { QRGB(205, 16,118),  "deeppink3" },
178
  { QRGB(139, 10, 80),  "deeppink4" },
179
  { QRGB(  0,191,255),  "deepskyblue" },
180
  { QRGB(  0,191,255),  "deepskyblue1" },
181
  { QRGB(  0,178,238),  "deepskyblue2" },
182
  { QRGB(  0,154,205),  "deepskyblue3" },
183
  { QRGB(  0,104,139),  "deepskyblue4" },
184
  { QRGB(105,105,105),  "dimgray" },
185
  { QRGB(105,105,105),  "dimgrey" },
186
  { QRGB( 30,144,255),  "dodgerblue" },
187
  { QRGB( 30,144,255),  "dodgerblue1" },
188
  { QRGB( 28,134,238),  "dodgerblue2" },
189
  { QRGB( 24,116,205),  "dodgerblue3" },
190
  { QRGB( 16, 78,139),  "dodgerblue4" },
191
  { QRGB(178, 34, 34),  "firebrick" },
192
  { QRGB(255, 48, 48),  "firebrick1" },
193
  { QRGB(238, 44, 44),  "firebrick2" },
194
  { QRGB(205, 38, 38),  "firebrick3" },
195
  { QRGB(139, 26, 26),  "firebrick4" },
196
  { QRGB(255,250,240),  "floralwhite" },
197
  { QRGB( 34,139, 34),  "forestgreen" },
198
  { QRGB(220,220,220),  "gainsboro" },
199
  { QRGB(248,248,255),  "ghostwhite" },
200
  { QRGB(255,215,  0),  "gold" },
201
  { QRGB(255,215,  0),  "gold1" },
202
  { QRGB(238,201,  0),  "gold2" },
203
  { QRGB(205,173,  0),  "gold3" },
204
  { QRGB(139,117,  0),  "gold4" },
205
  { QRGB(218,165, 32),  "goldenrod" },
206
  { QRGB(255,193, 37),  "goldenrod1" },
207
  { QRGB(238,180, 34),  "goldenrod2" },
208
  { QRGB(205,155, 29),  "goldenrod3" },
209
  { QRGB(139,105, 20),  "goldenrod4" },
210
  { QRGB(190,190,190),  "gray" },
211
  { QRGB(  0,  0,  0),  "gray0" },
212
  { QRGB(  3,  3,  3),  "gray1" },
213
  { QRGB( 26, 26, 26),  "gray10" },
214
  { QRGB(255,255,255),  "gray100" },
215
  { QRGB( 28, 28, 28),  "gray11" },
216
  { QRGB( 31, 31, 31),  "gray12" },
217
  { QRGB( 33, 33, 33),  "gray13" },
218
  { QRGB( 36, 36, 36),  "gray14" },
219
  { QRGB( 38, 38, 38),  "gray15" },
220
  { QRGB( 41, 41, 41),  "gray16" },
221
  { QRGB( 43, 43, 43),  "gray17" },
222
  { QRGB( 46, 46, 46),  "gray18" },
223
  { QRGB( 48, 48, 48),  "gray19" },
224
  { QRGB(  5,  5,  5),  "gray2" },
225
  { QRGB( 51, 51, 51),  "gray20" },
226
  { QRGB( 54, 54, 54),  "gray21" },
227
  { QRGB( 56, 56, 56),  "gray22" },
228
  { QRGB( 59, 59, 59),  "gray23" },
229
  { QRGB( 61, 61, 61),  "gray24" },
230
  { QRGB( 64, 64, 64),  "gray25" },
231
  { QRGB( 66, 66, 66),  "gray26" },
232
  { QRGB( 69, 69, 69),  "gray27" },
233
  { QRGB( 71, 71, 71),  "gray28" },
234
  { QRGB( 74, 74, 74),  "gray29" },
235
  { QRGB(  8,  8,  8),  "gray3" },
236
  { QRGB( 77, 77, 77),  "gray30" },
237
  { QRGB( 79, 79, 79),  "gray31" },
238
  { QRGB( 82, 82, 82),  "gray32" },
239
  { QRGB( 84, 84, 84),  "gray33" },
240
  { QRGB( 87, 87, 87),  "gray34" },
241
  { QRGB( 89, 89, 89),  "gray35" },
242
  { QRGB( 92, 92, 92),  "gray36" },
243
  { QRGB( 94, 94, 94),  "gray37" },
244
  { QRGB( 97, 97, 97),  "gray38" },
245
  { QRGB( 99, 99, 99),  "gray39" },
246
  { QRGB( 10, 10, 10),  "gray4" },
247
  { QRGB(102,102,102),  "gray40" },
248
  { QRGB(105,105,105),  "gray41" },
249
  { QRGB(107,107,107),  "gray42" },
250
  { QRGB(110,110,110),  "gray43" },
251
  { QRGB(112,112,112),  "gray44" },
252
  { QRGB(115,115,115),  "gray45" },
253
  { QRGB(117,117,117),  "gray46" },
254
  { QRGB(120,120,120),  "gray47" },
255
  { QRGB(122,122,122),  "gray48" },
256
  { QRGB(125,125,125),  "gray49" },
257
  { QRGB( 13, 13, 13),  "gray5" },
258
  { QRGB(127,127,127),  "gray50" },
259
  { QRGB(130,130,130),  "gray51" },
260
  { QRGB(133,133,133),  "gray52" },
261
  { QRGB(135,135,135),  "gray53" },
262
  { QRGB(138,138,138),  "gray54" },
263
  { QRGB(140,140,140),  "gray55" },
264
  { QRGB(143,143,143),  "gray56" },
265
  { QRGB(145,145,145),  "gray57" },
266
  { QRGB(148,148,148),  "gray58" },
267
  { QRGB(150,150,150),  "gray59" },
268
  { QRGB( 15, 15, 15),  "gray6" },
269
  { QRGB(153,153,153),  "gray60" },
270
  { QRGB(156,156,156),  "gray61" },
271
  { QRGB(158,158,158),  "gray62" },
272
  { QRGB(161,161,161),  "gray63" },
273
  { QRGB(163,163,163),  "gray64" },
274
  { QRGB(166,166,166),  "gray65" },
275
  { QRGB(168,168,168),  "gray66" },
276
  { QRGB(171,171,171),  "gray67" },
277
  { QRGB(173,173,173),  "gray68" },
278
  { QRGB(176,176,176),  "gray69" },
279
  { QRGB( 18, 18, 18),  "gray7" },
280
  { QRGB(179,179,179),  "gray70" },
281
  { QRGB(181,181,181),  "gray71" },
282
  { QRGB(184,184,184),  "gray72" },
283
  { QRGB(186,186,186),  "gray73" },
284
  { QRGB(189,189,189),  "gray74" },
285
  { QRGB(191,191,191),  "gray75" },
286
  { QRGB(194,194,194),  "gray76" },
287
  { QRGB(196,196,196),  "gray77" },
288
  { QRGB(199,199,199),  "gray78" },
289
  { QRGB(201,201,201),  "gray79" },
290
  { QRGB( 20, 20, 20),  "gray8" },
291
  { QRGB(204,204,204),  "gray80" },
292
  { QRGB(207,207,207),  "gray81" },
293
  { QRGB(209,209,209),  "gray82" },
294
  { QRGB(212,212,212),  "gray83" },
295
  { QRGB(214,214,214),  "gray84" },
296
  { QRGB(217,217,217),  "gray85" },
297
  { QRGB(219,219,219),  "gray86" },
298
  { QRGB(222,222,222),  "gray87" },
299
  { QRGB(224,224,224),  "gray88" },
300
  { QRGB(227,227,227),  "gray89" },
301
  { QRGB( 23, 23, 23),  "gray9" },
302
  { QRGB(229,229,229),  "gray90" },
303
  { QRGB(232,232,232),  "gray91" },
304
  { QRGB(235,235,235),  "gray92" },
305
  { QRGB(237,237,237),  "gray93" },
306
  { QRGB(240,240,240),  "gray94" },
307
  { QRGB(242,242,242),  "gray95" },
308
  { QRGB(245,245,245),  "gray96" },
309
  { QRGB(247,247,247),  "gray97" },
310
  { QRGB(250,250,250),  "gray98" },
311
  { QRGB(252,252,252),  "gray99" },
312
  { QRGB(  0,255,  0),  "green" },
313
  { QRGB(  0,255,  0),  "green1" },
314
  { QRGB(  0,238,  0),  "green2" },
315
  { QRGB(  0,205,  0),  "green3" },
316
  { QRGB(  0,139,  0),  "green4" },
317
  { QRGB(173,255, 47),  "greenyellow" },
318
  { QRGB(190,190,190),  "grey" },
319
  { QRGB(  0,  0,  0),  "grey0" },
320
  { QRGB(  3,  3,  3),  "grey1" },
321
  { QRGB( 26, 26, 26),  "grey10" },
322
  { QRGB(255,255,255),  "grey100" },
323
  { QRGB( 28, 28, 28),  "grey11" },
324
  { QRGB( 31, 31, 31),  "grey12" },
325
  { QRGB( 33, 33, 33),  "grey13" },
326
  { QRGB( 36, 36, 36),  "grey14" },
327
  { QRGB( 38, 38, 38),  "grey15" },
328
  { QRGB( 41, 41, 41),  "grey16" },
329
  { QRGB( 43, 43, 43),  "grey17" },
330
  { QRGB( 46, 46, 46),  "grey18" },
331
  { QRGB( 48, 48, 48),  "grey19" },
332
  { QRGB(  5,  5,  5),  "grey2" },
333
  { QRGB( 51, 51, 51),  "grey20" },
334
  { QRGB( 54, 54, 54),  "grey21" },
335
  { QRGB( 56, 56, 56),  "grey22" },
336
  { QRGB( 59, 59, 59),  "grey23" },
337
  { QRGB( 61, 61, 61),  "grey24" },
338
  { QRGB( 64, 64, 64),  "grey25" },
339
  { QRGB( 66, 66, 66),  "grey26" },
340
  { QRGB( 69, 69, 69),  "grey27" },
341
  { QRGB( 71, 71, 71),  "grey28" },
342
  { QRGB( 74, 74, 74),  "grey29" },
343
  { QRGB(  8,  8,  8),  "grey3" },
344
  { QRGB( 77, 77, 77),  "grey30" },
345
  { QRGB( 79, 79, 79),  "grey31" },
346
  { QRGB( 82, 82, 82),  "grey32" },
347
  { QRGB( 84, 84, 84),  "grey33" },
348
  { QRGB( 87, 87, 87),  "grey34" },
349
  { QRGB( 89, 89, 89),  "grey35" },
350
  { QRGB( 92, 92, 92),  "grey36" },
351
  { QRGB( 94, 94, 94),  "grey37" },
352
  { QRGB( 97, 97, 97),  "grey38" },
353
  { QRGB( 99, 99, 99),  "grey39" },
354
  { QRGB( 10, 10, 10),  "grey4" },
355
  { QRGB(102,102,102),  "grey40" },
356
  { QRGB(105,105,105),  "grey41" },
357
  { QRGB(107,107,107),  "grey42" },
358
  { QRGB(110,110,110),  "grey43" },
359
  { QRGB(112,112,112),  "grey44" },
360
  { QRGB(115,115,115),  "grey45" },
361
  { QRGB(117,117,117),  "grey46" },
362
  { QRGB(120,120,120),  "grey47" },
363
  { QRGB(122,122,122),  "grey48" },
364
  { QRGB(125,125,125),  "grey49" },
365
  { QRGB( 13, 13, 13),  "grey5" },
366
  { QRGB(127,127,127),  "grey50" },
367
  { QRGB(130,130,130),  "grey51" },
368
  { QRGB(133,133,133),  "grey52" },
369
  { QRGB(135,135,135),  "grey53" },
370
  { QRGB(138,138,138),  "grey54" },
371
  { QRGB(140,140,140),  "grey55" },
372
  { QRGB(143,143,143),  "grey56" },
373
  { QRGB(145,145,145),  "grey57" },
374
  { QRGB(148,148,148),  "grey58" },
375
  { QRGB(150,150,150),  "grey59" },
376
  { QRGB( 15, 15, 15),  "grey6" },
377
  { QRGB(153,153,153),  "grey60" },
378
  { QRGB(156,156,156),  "grey61" },
379
  { QRGB(158,158,158),  "grey62" },
380
  { QRGB(161,161,161),  "grey63" },
381
  { QRGB(163,163,163),  "grey64" },
382
  { QRGB(166,166,166),  "grey65" },
383
  { QRGB(168,168,168),  "grey66" },
384
  { QRGB(171,171,171),  "grey67" },
385
  { QRGB(173,173,173),  "grey68" },
386
  { QRGB(176,176,176),  "grey69" },
387
  { QRGB( 18, 18, 18),  "grey7" },
388
  { QRGB(179,179,179),  "grey70" },
389
  { QRGB(181,181,181),  "grey71" },
390
  { QRGB(184,184,184),  "grey72" },
391
  { QRGB(186,186,186),  "grey73" },
392
  { QRGB(189,189,189),  "grey74" },
393
  { QRGB(191,191,191),  "grey75" },
394
  { QRGB(194,194,194),  "grey76" },
395
  { QRGB(196,196,196),  "grey77" },
396
  { QRGB(199,199,199),  "grey78" },
397
  { QRGB(201,201,201),  "grey79" },
398
  { QRGB( 20, 20, 20),  "grey8" },
399
  { QRGB(204,204,204),  "grey80" },
400
  { QRGB(207,207,207),  "grey81" },
401
  { QRGB(209,209,209),  "grey82" },
402
  { QRGB(212,212,212),  "grey83" },
403
  { QRGB(214,214,214),  "grey84" },
404
  { QRGB(217,217,217),  "grey85" },
405
  { QRGB(219,219,219),  "grey86" },
406
  { QRGB(222,222,222),  "grey87" },
407
  { QRGB(224,224,224),  "grey88" },
408
  { QRGB(227,227,227),  "grey89" },
409
  { QRGB( 23, 23, 23),  "grey9" },
410
  { QRGB(229,229,229),  "grey90" },
411
  { QRGB(232,232,232),  "grey91" },
412
  { QRGB(235,235,235),  "grey92" },
413
  { QRGB(237,237,237),  "grey93" },
414
  { QRGB(240,240,240),  "grey94" },
415
  { QRGB(242,242,242),  "grey95" },
416
  { QRGB(245,245,245),  "grey96" },
417
  { QRGB(247,247,247),  "grey97" },
418
  { QRGB(250,250,250),  "grey98" },
419
  { QRGB(252,252,252),  "grey99" },
420
  { QRGB(240,255,240),  "honeydew" },
421
  { QRGB(240,255,240),  "honeydew1" },
422
  { QRGB(224,238,224),  "honeydew2" },
423
  { QRGB(193,205,193),  "honeydew3" },
424
  { QRGB(131,139,131),  "honeydew4" },
425
  { QRGB(255,105,180),  "hotpink" },
426
  { QRGB(255,110,180),  "hotpink1" },
427
  { QRGB(238,106,167),  "hotpink2" },
428
  { QRGB(205, 96,144),  "hotpink3" },
429
  { QRGB(139, 58, 98),  "hotpink4" },
430
  { QRGB(205, 92, 92),  "indianred" },
431
  { QRGB(255,106,106),  "indianred1" },
432
  { QRGB(238, 99, 99),  "indianred2" },
433
  { QRGB(205, 85, 85),  "indianred3" },
434
  { QRGB(139, 58, 58),  "indianred4" },
435
  { QRGB(255,255,240),  "ivory" },
436
  { QRGB(255,255,240),  "ivory1" },
437
  { QRGB(238,238,224),  "ivory2" },
438
  { QRGB(205,205,193),  "ivory3" },
439
  { QRGB(139,139,131),  "ivory4" },
440
  { QRGB(240,230,140),  "khaki" },
441
  { QRGB(255,246,143),  "khaki1" },
442
  { QRGB(238,230,133),  "khaki2" },
443
  { QRGB(205,198,115),  "khaki3" },
444
  { QRGB(139,134, 78),  "khaki4" },
445
  { QRGB(230,230,250),  "lavender" },
446
  { QRGB(255,240,245),  "lavenderblush" },
447
  { QRGB(255,240,245),  "lavenderblush1" },
448
  { QRGB(238,224,229),  "lavenderblush2" },
449
  { QRGB(205,193,197),  "lavenderblush3" },
450
  { QRGB(139,131,134),  "lavenderblush4" },
451
  { QRGB(124,252,  0),  "lawngreen" },
452
  { QRGB(255,250,205),  "lemonchiffon" },
453
  { QRGB(255,250,205),  "lemonchiffon1" },
454
  { QRGB(238,233,191),  "lemonchiffon2" },
455
  { QRGB(205,201,165),  "lemonchiffon3" },
456
  { QRGB(139,137,112),  "lemonchiffon4" },
457
  { QRGB(173,216,230),  "lightblue" },
458
  { QRGB(191,239,255),  "lightblue1" },
459
  { QRGB(178,223,238),  "lightblue2" },
460
  { QRGB(154,192,205),  "lightblue3" },
461
  { QRGB(104,131,139),  "lightblue4" },
462
  { QRGB(240,128,128),  "lightcoral" },
463
  { QRGB(224,255,255),  "lightcyan" },
464
  { QRGB(224,255,255),  "lightcyan1" },
465
  { QRGB(209,238,238),  "lightcyan2" },
466
  { QRGB(180,205,205),  "lightcyan3" },
467
  { QRGB(122,139,139),  "lightcyan4" },
468
  { QRGB(238,221,130),  "lightgoldenrod" },
469
  { QRGB(255,236,139),  "lightgoldenrod1" },
470
  { QRGB(238,220,130),  "lightgoldenrod2" },
471
  { QRGB(205,190,112),  "lightgoldenrod3" },
472
  { QRGB(139,129, 76),  "lightgoldenrod4" },
473
  { QRGB(250,250,210),  "lightgoldenrodyellow" },
474
  { QRGB(211,211,211),  "lightgray" },
475
  { QRGB(144,238,144),  "lightgreen" },
476
  { QRGB(211,211,211),  "lightgrey" },
477
  { QRGB(255,182,193),  "lightpink" },
478
  { QRGB(255,174,185),  "lightpink1" },
479
  { QRGB(238,162,173),  "lightpink2" },
480
  { QRGB(205,140,149),  "lightpink3" },
481
  { QRGB(139, 95,101),  "lightpink4" },
482
  { QRGB(255,160,122),  "lightsalmon" },
483
  { QRGB(255,160,122),  "lightsalmon1" },
484
  { QRGB(238,149,114),  "lightsalmon2" },
485
  { QRGB(205,129, 98),  "lightsalmon3" },
486
  { QRGB(139, 87, 66),  "lightsalmon4" },
487
  { QRGB( 32,178,170),  "lightseagreen" },
488
  { QRGB(135,206,250),  "lightskyblue" },
489
  { QRGB(176,226,255),  "lightskyblue1" },
490
  { QRGB(164,211,238),  "lightskyblue2" },
491
  { QRGB(141,182,205),  "lightskyblue3" },
492
  { QRGB( 96,123,139),  "lightskyblue4" },
493
  { QRGB(132,112,255),  "lightslateblue" },
494
  { QRGB(119,136,153),  "lightslategray" },
495
  { QRGB(119,136,153),  "lightslategrey" },
496
  { QRGB(176,196,222),  "lightsteelblue" },
497
  { QRGB(202,225,255),  "lightsteelblue1" },
498
  { QRGB(188,210,238),  "lightsteelblue2" },
499
  { QRGB(162,181,205),  "lightsteelblue3" },
500
  { QRGB(110,123,139),  "lightsteelblue4" },
501
  { QRGB(255,255,224),  "lightyellow" },
502
  { QRGB(255,255,224),  "lightyellow1" },
503
  { QRGB(238,238,209),  "lightyellow2" },
504
  { QRGB(205,205,180),  "lightyellow3" },
505
  { QRGB(139,139,122),  "lightyellow4" },
506
  { QRGB( 50,205, 50),  "limegreen" },
507
  { QRGB(250,240,230),  "linen" },
508
  { QRGB(255,  0,255),  "magenta" },
509
  { QRGB(255,  0,255),  "magenta1" },
510
  { QRGB(238,  0,238),  "magenta2" },
511
  { QRGB(205,  0,205),  "magenta3" },
512
  { QRGB(139,  0,139),  "magenta4" },
513
  { QRGB(176, 48, 96),  "maroon" },
514
  { QRGB(255, 52,179),  "maroon1" },
515
  { QRGB(238, 48,167),  "maroon2" },
516
  { QRGB(205, 41,144),  "maroon3" },
517
  { QRGB(139, 28, 98),  "maroon4" },
518
  { QRGB(102,205,170),  "mediumaquamarine" },
519
  { QRGB(  0,  0,205),  "mediumblue" },
520
  { QRGB(186, 85,211),  "mediumorchid" },
521
  { QRGB(224,102,255),  "mediumorchid1" },
522
  { QRGB(209, 95,238),  "mediumorchid2" },
523
  { QRGB(180, 82,205),  "mediumorchid3" },
524
  { QRGB(122, 55,139),  "mediumorchid4" },
525
  { QRGB(147,112,219),  "mediumpurple" },
526
  { QRGB(171,130,255),  "mediumpurple1" },
527
  { QRGB(159,121,238),  "mediumpurple2" },
528
  { QRGB(137,104,205),  "mediumpurple3" },
529
  { QRGB( 93, 71,139),  "mediumpurple4" },
530
  { QRGB( 60,179,113),  "mediumseagreen" },
531
  { QRGB(123,104,238),  "mediumslateblue" },
532
  { QRGB(  0,250,154),  "mediumspringgreen" },
533
  { QRGB( 72,209,204),  "mediumturquoise" },
534
  { QRGB(199, 21,133),  "mediumvioletred" },
535
  { QRGB( 25, 25,112),  "midnightblue" },
536
  { QRGB(245,255,250),  "mintcream" },
537
  { QRGB(255,228,225),  "mistyrose" },
538
  { QRGB(255,228,225),  "mistyrose1" },
539
  { QRGB(238,213,210),  "mistyrose2" },
540
  { QRGB(205,183,181),  "mistyrose3" },
541
  { QRGB(139,125,123),  "mistyrose4" },
542
  { QRGB(255,228,181),  "moccasin" },
543
  { QRGB(255,222,173),  "navajowhite" },
544
  { QRGB(255,222,173),  "navajowhite1" },
545
  { QRGB(238,207,161),  "navajowhite2" },
546
  { QRGB(205,179,139),  "navajowhite3" },
547
  { QRGB(139,121, 94),  "navajowhite4" },
548
  { QRGB(  0,  0,128),  "navy" },
549
  { QRGB(  0,  0,128),  "navyblue" },
550
  { QRGB(253,245,230),  "oldlace" },
551
  { QRGB(107,142, 35),  "olivedrab" },
552
  { QRGB(192,255, 62),  "olivedrab1" },
553
  { QRGB(179,238, 58),  "olivedrab2" },
554
  { QRGB(154,205, 50),  "olivedrab3" },
555
  { QRGB(105,139, 34),  "olivedrab4" },
556
  { QRGB(255,165,  0),  "orange" },
557
  { QRGB(255,165,  0),  "orange1" },
558
  { QRGB(238,154,  0),  "orange2" },
559
  { QRGB(205,133,  0),  "orange3" },
560
  { QRGB(139, 90,  0),  "orange4" },
561
  { QRGB(255, 69,  0),  "orangered" },
562
  { QRGB(255, 69,  0),  "orangered1" },
563
  { QRGB(238, 64,  0),  "orangered2" },
564
  { QRGB(205, 55,  0),  "orangered3" },
565
  { QRGB(139, 37,  0),  "orangered4" },
566
  { QRGB(218,112,214),  "orchid" },
567
  { QRGB(255,131,250),  "orchid1" },
568
  { QRGB(238,122,233),  "orchid2" },
569
  { QRGB(205,105,201),  "orchid3" },
570
  { QRGB(139, 71,137),  "orchid4" },
571
  { QRGB(238,232,170),  "palegoldenrod" },
572
  { QRGB(152,251,152),  "palegreen" },
573
  { QRGB(154,255,154),  "palegreen1" },
574
  { QRGB(144,238,144),  "palegreen2" },
575
  { QRGB(124,205,124),  "palegreen3" },
576
  { QRGB( 84,139, 84),  "palegreen4" },
577
  { QRGB(175,238,238),  "paleturquoise" },
578
  { QRGB(187,255,255),  "paleturquoise1" },
579
  { QRGB(174,238,238),  "paleturquoise2" },
580
  { QRGB(150,205,205),  "paleturquoise3" },
581
  { QRGB(102,139,139),  "paleturquoise4" },
582
  { QRGB(219,112,147),  "palevioletred" },
583
  { QRGB(255,130,171),  "palevioletred1" },
584
  { QRGB(238,121,159),  "palevioletred2" },
585
  { QRGB(205,104,137),  "palevioletred3" },
586
  { QRGB(139, 71, 93),  "palevioletred4" },
587
  { QRGB(255,239,213),  "papayawhip" },
588
  { QRGB(255,218,185),  "peachpuff" },
589
  { QRGB(255,218,185),  "peachpuff1" },
590
  { QRGB(238,203,173),  "peachpuff2" },
591
  { QRGB(205,175,149),  "peachpuff3" },
592
  { QRGB(139,119,101),  "peachpuff4" },
593
  { QRGB(205,133, 63),  "peru" },
594
  { QRGB(255,192,203),  "pink" },
595
  { QRGB(255,181,197),  "pink1" },
596
  { QRGB(238,169,184),  "pink2" },
597
  { QRGB(205,145,158),  "pink3" },
598
  { QRGB(139, 99,108),  "pink4" },
599
  { QRGB(221,160,221),  "plum" },
600
  { QRGB(255,187,255),  "plum1" },
601
  { QRGB(238,174,238),  "plum2" },
602
  { QRGB(205,150,205),  "plum3" },
603
  { QRGB(139,102,139),  "plum4" },
604
  { QRGB(176,224,230),  "powderblue" },
605
  { QRGB(160, 32,240),  "purple" },
606
  { QRGB(155, 48,255),  "purple1" },
607
  { QRGB(145, 44,238),  "purple2" },
608
  { QRGB(125, 38,205),  "purple3" },
609
  { QRGB( 85, 26,139),  "purple4" },
610
  { QRGB(255,  0,  0),  "red" },
611
  { QRGB(255,  0,  0),  "red1" },
612
  { QRGB(238,  0,  0),  "red2" },
613
  { QRGB(205,  0,  0),  "red3" },
614
  { QRGB(139,  0,  0),  "red4" },
615
  { QRGB(188,143,143),  "rosybrown" },
616
  { QRGB(255,193,193),  "rosybrown1" },
617
  { QRGB(238,180,180),  "rosybrown2" },
618
  { QRGB(205,155,155),  "rosybrown3" },
619
  { QRGB(139,105,105),  "rosybrown4" },
620
  { QRGB( 65,105,225),  "royalblue" },
621
  { QRGB( 72,118,255),  "royalblue1" },
622
  { QRGB( 67,110,238),  "royalblue2" },
623
  { QRGB( 58, 95,205),  "royalblue3" },
624
  { QRGB( 39, 64,139),  "royalblue4" },
625
  { QRGB(139, 69, 19),  "saddlebrown" },
626
  { QRGB(250,128,114),  "salmon" },
627
  { QRGB(255,140,105),  "salmon1" },
628
  { QRGB(238,130, 98),  "salmon2" },
629
  { QRGB(205,112, 84),  "salmon3" },
630
  { QRGB(139, 76, 57),  "salmon4" },
631
  { QRGB(244,164, 96),  "sandybrown" },
632
  { QRGB( 46,139, 87),  "seagreen" },
633
  { QRGB( 84,255,159),  "seagreen1" },
634
  { QRGB( 78,238,148),  "seagreen2" },
635
  { QRGB( 67,205,128),  "seagreen3" },
636
  { QRGB( 46,139, 87),  "seagreen4" },
637
  { QRGB(255,245,238),  "seashell" },
638
  { QRGB(255,245,238),  "seashell1" },
639
  { QRGB(238,229,222),  "seashell2" },
640
  { QRGB(205,197,191),  "seashell3" },
641
  { QRGB(139,134,130),  "seashell4" },
642
  { QRGB(160, 82, 45),  "sienna" },
643
  { QRGB(255,130, 71),  "sienna1" },
644
  { QRGB(238,121, 66),  "sienna2" },
645
  { QRGB(205,104, 57),  "sienna3" },
646
  { QRGB(139, 71, 38),  "sienna4" },
647
  { QRGB(135,206,235),  "skyblue" },
648
  { QRGB(135,206,255),  "skyblue1" },
649
  { QRGB(126,192,238),  "skyblue2" },
650
  { QRGB(108,166,205),  "skyblue3" },
651
  { QRGB( 74,112,139),  "skyblue4" },
652
  { QRGB(106, 90,205),  "slateblue" },
653
  { QRGB(131,111,255),  "slateblue1" },
654
  { QRGB(122,103,238),  "slateblue2" },
655
  { QRGB(105, 89,205),  "slateblue3" },
656
  { QRGB( 71, 60,139),  "slateblue4" },
657
  { QRGB(112,128,144),  "slategray" },
658
  { QRGB(198,226,255),  "slategray1" },
659
  { QRGB(185,211,238),  "slategray2" },
660
  { QRGB(159,182,205),  "slategray3" },
661
  { QRGB(108,123,139),  "slategray4" },
662
  { QRGB(112,128,144),  "slategrey" },
663
  { QRGB(255,250,250),  "snow" },
664
  { QRGB(255,250,250),  "snow1" },
665
  { QRGB(238,233,233),  "snow2" },
666
  { QRGB(205,201,201),  "snow3" },
667
  { QRGB(139,137,137),  "snow4" },
668
  { QRGB(  0,255,127),  "springgreen" },
669
  { QRGB(  0,255,127),  "springgreen1" },
670
  { QRGB(  0,238,118),  "springgreen2" },
671
  { QRGB(  0,205,102),  "springgreen3" },
672
  { QRGB(  0,139, 69),  "springgreen4" },
673
  { QRGB( 70,130,180),  "steelblue" },
674
  { QRGB( 99,184,255),  "steelblue1" },
675
  { QRGB( 92,172,238),  "steelblue2" },
676
  { QRGB( 79,148,205),  "steelblue3" },
677
  { QRGB( 54,100,139),  "steelblue4" },
678
  { QRGB(210,180,140),  "tan" },
679
  { QRGB(255,165, 79),  "tan1" },
680
  { QRGB(238,154, 73),  "tan2" },
681
  { QRGB(205,133, 63),  "tan3" },
682
  { QRGB(139, 90, 43),  "tan4" },
683
  { QRGB(216,191,216),  "thistle" },
684
  { QRGB(255,225,255),  "thistle1" },
685
  { QRGB(238,210,238),  "thistle2" },
686
  { QRGB(205,181,205),  "thistle3" },
687
  { QRGB(139,123,139),  "thistle4" },
688
  { QRGB(255, 99, 71),  "tomato" },
689
  { QRGB(255, 99, 71),  "tomato1" },
690
  { QRGB(238, 92, 66),  "tomato2" },
691
  { QRGB(205, 79, 57),  "tomato3" },
692
  { QRGB(139, 54, 38),  "tomato4" },
693
  { QRGB( 64,224,208),  "turquoise" },
694
  { QRGB(  0,245,255),  "turquoise1" },
695
  { QRGB(  0,229,238),  "turquoise2" },
696
  { QRGB(  0,197,205),  "turquoise3" },
697
  { QRGB(  0,134,139),  "turquoise4" },
698
  { QRGB(238,130,238),  "violet" },
699
  { QRGB(208, 32,144),  "violetred" },
700
  { QRGB(255, 62,150),  "violetred1" },
701
  { QRGB(238, 58,140),  "violetred2" },
702
  { QRGB(205, 50,120),  "violetred3" },
703
  { QRGB(139, 34, 82),  "violetred4" },
704
  { QRGB(245,222,179),  "wheat" },
705
  { QRGB(255,231,186),  "wheat1" },
706
  { QRGB(238,216,174),  "wheat2" },
707
  { QRGB(205,186,150),  "wheat3" },
708
  { QRGB(139,126,102),  "wheat4" },
709
  { QRGB(255,255,255),  "white" },
710
  { QRGB(245,245,245),  "whitesmoke" },
711
  { QRGB(255,255,  0),  "yellow" },
712
  { QRGB(255,255,  0),  "yellow1" },
713
  { QRGB(238,238,  0),  "yellow2" },
714
  { QRGB(205,205,  0),  "yellow3" },
715
  { QRGB(139,139,  0),  "yellow4" },
716
  { QRGB(154,205, 50),  "yellowgreen" } };
717
718
719
inline bool operator<(const char *name, const XPMRGBData &data)
720
0
{ return qstrcmp(name, data.name) < 0; }
721
inline bool operator<(const XPMRGBData &data, const char *name)
722
0
{ return qstrcmp(data.name, name) < 0; }
723
724
static inline std::optional<QRgb> qt_get_named_xpm_rgb(const char *name_no_space)
725
0
{
726
0
    const XPMRGBData *r = std::lower_bound(xpmRgbTbl, xpmRgbTbl + xpmRgbTblSize, name_no_space);
727
0
    if ((r != xpmRgbTbl + xpmRgbTblSize) && !(name_no_space < *r))
728
0
        return r->value;
729
0
    return {};
730
0
}
731
732
/*****************************************************************************
733
  Misc. utility functions
734
 *****************************************************************************/
735
static QString fbname(const QString &fileName) // get file basename (sort of)
736
0
{
737
0
    QString s = fileName;
738
0
    if (!s.isEmpty()) {
739
0
        int i = qMax(s.lastIndexOf(u'/'), s.lastIndexOf(u'\\'));
740
0
        if (i < 0)
741
0
            i = 0;
742
0
        auto checkChar = [](QChar ch) -> bool {
743
0
            uchar uc = ch.unicode();
744
0
            return isAsciiLetterOrNumber(uc) || uc == '_';
745
0
        };
746
0
        int start = -1;
747
0
        for (; i < s.size(); ++i) {
748
0
            if (checkChar(s.at(i))) {
749
0
                start = i;
750
0
            } else if (start > 0)
751
0
                break;
752
0
        }
753
0
        if (start < 0)
754
0
            s.clear();
755
0
        else
756
0
            s = s.mid(start, i - start);
757
0
    }
758
0
    if (s.isEmpty())
759
0
        s = QString::fromLatin1("dummy");
760
0
    return s;
761
0
}
762
763
// Skip until ", read until the next ", return the rest in *buf
764
// Returns false on error, true on success
765
766
static bool read_xpm_string(QByteArray &buf, QIODevice *d, const char * const *source, int &index,
767
                            QByteArray &state)
768
0
{
769
0
    if (source) {
770
0
        buf = source[index++];
771
0
        return true;
772
0
    }
773
774
0
    buf = "";
775
0
    bool gotQuote = false;
776
0
    int offset = 0;
777
0
    forever {
778
0
        if (offset == state.size() || state.isEmpty()) {
779
0
            char buf[2048];
780
0
            qint64 bytesRead = d->read(buf, sizeof(buf));
781
0
            if (bytesRead <= 0)
782
0
                return false;
783
0
            state = QByteArray(buf, int(bytesRead));
784
0
            offset = 0;
785
0
        }
786
787
0
        if (!gotQuote) {
788
0
            if (state.at(offset++) == '"')
789
0
                gotQuote = true;
790
0
        } else {
791
0
            char c = state.at(offset++);
792
0
            if (c == '"')
793
0
                break;
794
0
            buf += c;
795
0
        }
796
0
    }
797
0
    state.remove(0, offset);
798
0
    return true;
799
0
}
800
801
// Tests if the given prefix can be the start of an XPM color specification
802
803
static bool is_xpm_color_spec_prefix(const QByteArray& prefix)
804
0
{
805
0
    return prefix == "c" ||
806
0
           prefix == "g" ||
807
0
           prefix == "g4" ||
808
0
           prefix == "m" ||
809
0
           prefix == "s";
810
0
}
811
812
// Reads XPM header.
813
814
static bool read_xpm_header(
815
    QIODevice *device, const char * const * source, int& index, QByteArray &state,
816
    int *cpp, int *ncols, int *w, int *h)
817
0
{
818
0
    QByteArray buf(200, 0);
819
820
0
    if (!read_xpm_string(buf, device, source, index, state))
821
0
        return false;
822
823
#ifdef Q_CC_MSVC
824
        if (sscanf_s(buf, "%d %d %d %d", w, h, ncols, cpp) < 4)
825
#else
826
0
    if (sscanf(buf, "%d %d %d %d", w, h, ncols, cpp) < 4)
827
0
#endif
828
0
        return false;                                        // < 4 numbers parsed
829
830
0
    if (*w <= 0 || *w > 32767 || *h <= 0 || *h > 32767 || *ncols <= 0 || *ncols > (64 * 64 * 64 * 64) || *cpp <= 0 || *cpp > 15)
831
0
        return false;                                        // failed sanity check
832
833
0
    return true;
834
0
}
835
836
// Reads XPM body (color information & pixels).
837
838
static bool read_xpm_body(
839
    QIODevice *device, const char * const * source, int& index, QByteArray& state,
840
    int cpp, int ncols, int w, int h, QImage& image)
841
0
{
842
0
    QByteArray buf(200, 0);
843
0
    int i;
844
845
0
    if (cpp < 0 || cpp > 15)
846
0
        return false;
847
848
    // For > 256 colors, we delay creation of the image until
849
    // after we have read the color specifications, so that we can
850
    // create it in correct format (Format_RGB32 vs Format_ARGB32,
851
    // depending on absence or presence of "c none", respectively)
852
0
    if (ncols <= 256) {
853
0
        if (!QImageIOHandler::allocateImage(QSize(w, h), QImage::Format_Indexed8, &image))
854
0
            return false;
855
0
        image.setColorCount(ncols);
856
0
    }
857
858
0
    QMap<quint64, int> colorMap;
859
0
    int currentColor;
860
0
    bool hasTransparency = false;
861
862
0
    for(currentColor=0; currentColor < ncols; ++currentColor) {
863
0
        if (!read_xpm_string(buf, device, source, index, state)) {
864
0
            qCWarning(lcImageIo, "XPM color specification missing");
865
0
            return false;
866
0
        }
867
0
        QByteArray index;
868
0
        index = buf.left(cpp);
869
0
        buf = buf.mid(cpp).simplified().trimmed().toLower();
870
0
        QList<QByteArray> tokens = buf.split(' ');
871
0
        i = tokens.indexOf("c");
872
0
        if (i < 0)
873
0
            i = tokens.indexOf("g");
874
0
        if (i < 0)
875
0
            i = tokens.indexOf("g4");
876
0
        if (i < 0)
877
0
            i = tokens.indexOf("m");
878
0
        if (i < 0) {
879
0
            qCWarning(lcImageIo, "XPM color specification is missing: %s", buf.constData());
880
0
            return false;        // no c/g/g4/m specification at all
881
0
        }
882
0
        QByteArray color;
883
0
        while ((++i < tokens.size()) && !is_xpm_color_spec_prefix(tokens.at(i))) {
884
0
            color.append(tokens.at(i));
885
0
        }
886
0
        if (color.isEmpty()) {
887
0
            qCWarning(lcImageIo, "XPM color value is missing from specification: %s", buf.constData());
888
0
            return false;        // no color value
889
0
        }
890
0
        buf = color;
891
0
        if (buf == "none") {
892
0
            hasTransparency = true;
893
0
            int transparentColor = currentColor;
894
0
            if (ncols <= 256) {
895
0
                image.setColor(transparentColor, 0);
896
0
                colorMap.insert(xpmHash(QLatin1StringView(index.constData())), transparentColor);
897
0
            } else {
898
0
                colorMap.insert(xpmHash(QLatin1StringView(index.constData())), 0);
899
0
            }
900
0
        } else {
901
0
            QRgb c_rgb = 0;
902
0
            if (((buf.size()-1) % 3) && (buf[0] == '#')) {
903
0
                buf.truncate(((buf.size()-1) / 4 * 3) + 1); // remove alpha channel left by imagemagick
904
0
            }
905
0
            if (buf[0] == '#') {
906
0
                c_rgb = qt_get_hex_rgb(buf).value_or(0);
907
0
            } else {
908
0
                c_rgb = qt_get_named_xpm_rgb(buf).value_or(0);
909
0
            }
910
0
            if (ncols <= 256) {
911
0
                image.setColor(currentColor, 0xff000000 | c_rgb);
912
0
                colorMap.insert(xpmHash(QLatin1StringView(index.constData())), currentColor);
913
0
            } else {
914
0
                colorMap.insert(xpmHash(QLatin1StringView(index.constData())), 0xff000000 | c_rgb);
915
0
            }
916
0
        }
917
0
    }
918
919
0
    if (ncols > 256) {
920
        // Now we can create 32-bit image of appropriate format
921
0
        QImage::Format format = hasTransparency ?
922
0
                                QImage::Format_ARGB32 : QImage::Format_RGB32;
923
0
        if (!QImageIOHandler::allocateImage(QSize(w, h), format, &image))
924
0
            return false;
925
0
    }
926
927
    // Read pixels
928
0
    for(int y=0; y<h; y++) {
929
0
        if (!read_xpm_string(buf, device, source, index, state)) {
930
0
            qCWarning(lcImageIo, "XPM pixels missing on image line %d", y);
931
0
            return false;
932
0
        }
933
0
        if (image.depth() == 8) {
934
0
            uchar *p = image.scanLine(y);
935
0
            uchar *d = (uchar *)buf.data();
936
0
            uchar *end = d + buf.size();
937
0
            int x;
938
0
            if (cpp == 1) {
939
0
                char b[2];
940
0
                b[1] = '\0';
941
0
                for (x=0; x<w && d<end; x++) {
942
0
                    b[0] = *d++;
943
0
                    *p++ = (uchar)colorMap[xpmHash(b)];
944
0
                }
945
0
            } else {
946
0
                char b[16];
947
0
                b[cpp] = '\0';
948
0
                for (x = 0; x < w && d + cpp <= end; x++) {
949
0
                    memcpy(b, (char *)d, cpp);
950
0
                    *p++ = (uchar)colorMap[xpmHash(b)];
951
0
                    d += cpp;
952
0
                }
953
0
            }
954
            // avoid uninitialized memory for malformed xpms
955
0
            if (x < w) {
956
0
                qCWarning(lcImageIo, "XPM pixels missing on image line %d (possibly a C++ trigraph).", y);
957
0
                memset(p, 0, w - x);
958
0
            }
959
0
        } else {
960
0
            QRgb *p = (QRgb*)image.scanLine(y);
961
0
            uchar *d = (uchar *)buf.data();
962
0
            uchar *end = d + buf.size();
963
0
            int x;
964
0
            char b[16];
965
0
            b[cpp] = '\0';
966
0
            for (x = 0; x < w && d + cpp <= end; x++) {
967
0
                memcpy(b, (char *)d, cpp);
968
0
                *p++ = (QRgb)colorMap[xpmHash(b)];
969
0
                d += cpp;
970
0
            }
971
            // avoid uninitialized memory for malformed xpms
972
0
            if (x < w) {
973
0
                qCWarning(lcImageIo, "XPM pixels missing on image line %d (possibly a C++ trigraph).", y);
974
0
                memset(p, 0, (w - x)*4);
975
0
            }
976
0
        }
977
0
    }
978
979
0
    if (device) {
980
        // Rewind unused characters, and skip to the end of the XPM struct.
981
0
        for (int i = state.size() - 1; i >= 0; --i)
982
0
            device->ungetChar(state[i]);
983
0
        char c;
984
0
        while (device->getChar(&c) && c != ';') {}
985
0
        while (device->getChar(&c) && c != '\n') {}
986
0
    }
987
0
    return true;
988
0
}
989
990
//
991
// INTERNAL
992
//
993
// Reads an .xpm from either the QImageIO or from the QString *.
994
// One of the two HAS to be 0, the other one is used.
995
//
996
997
bool qt_read_xpm_image_or_array(QIODevice *device, const char * const * source, QImage &image)
998
0
{
999
0
    if (!source)
1000
0
        return true;
1001
1002
0
    QByteArray buf(200, 0);
1003
0
    QByteArray state;
1004
1005
0
    int cpp, ncols, w, h, index = 0;
1006
1007
0
    if (device) {
1008
        // "/* XPM */"
1009
0
        int readBytes;
1010
0
        if ((readBytes = device->readLine(buf.data(), buf.size())) < 0)
1011
0
            return false;
1012
1013
0
        static constexpr auto matcher = qMakeStaticByteArrayMatcher("/* XPM");
1014
1015
0
        if (matcher.indexIn(buf) != 0) {
1016
0
            while (readBytes > 0) {
1017
0
                device->ungetChar(buf.at(readBytes - 1));
1018
0
                --readBytes;
1019
0
            }
1020
0
            return false;
1021
0
        }// bad magic
1022
0
    }
1023
1024
0
    if (!read_xpm_header(device, source, index, state, &cpp, &ncols, &w, &h))
1025
0
        return false;
1026
1027
0
    return read_xpm_body(device, source, index, state, cpp, ncols, w, h, image);
1028
0
}
1029
1030
namespace {
1031
template <size_t N>
1032
struct CharBuffer : std::array<char, N>
1033
{
1034
0
    CharBuffer() {} // avoid value-initializing the whole array
1035
};
1036
}
1037
1038
static const char* xpm_color_name(int cpp, int index, CharBuffer<5> && returnable = {})
1039
0
{
1040
0
    static const char code[] = ".#abcdefghijklmnopqrstuvwxyzABCD"
1041
0
                               "EFGHIJKLMNOPQRSTUVWXYZ0123456789";
1042
    // cpp is limited to 4 and index is limited to 64^cpp
1043
0
    if (cpp > 1) {
1044
0
        if (cpp > 2) {
1045
0
            if (cpp > 3) {
1046
0
                returnable[4] = '\0';
1047
0
                returnable[3] = code[index % 64];
1048
0
                index /= 64;
1049
0
            } else
1050
0
                returnable[3] = '\0';
1051
0
            returnable[2] = code[index % 64];
1052
0
            index /= 64;
1053
0
        } else
1054
0
            returnable[2] = '\0';
1055
        // the following 4 lines are a joke!
1056
0
        if (index == 0)
1057
0
            index = 64*44+21;
1058
0
        else if (index == 64*44+21)
1059
0
            index = 0;
1060
0
        returnable[1] = code[index % 64];
1061
0
        index /= 64;
1062
0
    } else
1063
0
        returnable[1] = '\0';
1064
0
    returnable[0] = code[index];
1065
1066
0
    return returnable.data();
1067
0
}
1068
1069
1070
// write XPM image data
1071
static bool write_xpm_image(const QImage &sourceImage, QIODevice *device, const QString &fileName)
1072
0
{
1073
0
    if (!device->isWritable())
1074
0
        return false;
1075
1076
0
    QImage image;
1077
0
    if (sourceImage.format() != QImage::Format_RGB32 && sourceImage.format() != QImage::Format_ARGB32 && sourceImage.format() != QImage::Format_ARGB32_Premultiplied)
1078
0
        image = sourceImage.convertToFormat(QImage::Format_RGB32);
1079
0
    else
1080
0
        image = sourceImage;
1081
1082
0
#ifdef __cpp_lib_memory_resource
1083
0
    char buffer[1024];
1084
0
    std::pmr::monotonic_buffer_resource res{&buffer, sizeof buffer};
1085
0
    std::pmr::map<QRgb, int> colorMap(&res);
1086
#else
1087
    std::map<QRgb, int> colorMap;
1088
#endif
1089
1090
0
    const int w = image.width();
1091
0
    const int h = image.height();
1092
0
    int ncolors = 0;
1093
1094
    // build color table
1095
0
    for (int y = 0; y < h; ++y) {
1096
0
        const QRgb *yp = reinterpret_cast<const QRgb *>(image.constScanLine(y));
1097
0
        for (int x = 0; x < w; ++x) {
1098
0
            const auto [it, inserted] = colorMap.try_emplace(yp[x], ncolors);
1099
0
            if (inserted)
1100
0
                ++ncolors;
1101
0
        }
1102
0
    }
1103
1104
    // number of 64-bit characters per pixel needed to encode all colors
1105
0
    int cpp = 1;
1106
0
    for (int k = 64; ncolors > k; k *= 64) {
1107
0
        ++cpp;
1108
        // limit to 4 characters per pixel
1109
        // 64^4 colors is enough for a 4096x4096 image
1110
0
         if (cpp > 4) {
1111
0
             qCWarning(lcImageIo, "Qt does not support writing XPM images with more than "
1112
0
                       "64^4 colors (requested: %d colors).", ncolors);
1113
0
             return false;
1114
0
         }
1115
0
    }
1116
1117
    // write header
1118
0
    QTextStream s(device);
1119
0
    s << "/* XPM */" << Qt::endl
1120
0
      << "static char *" << fbname(fileName) << "[]={" << Qt::endl
1121
0
      << '\"' << w << ' ' << h << ' ' << ncolors << ' ' << cpp << '\"';
1122
1123
    // write palette
1124
0
    for (const auto &[color, index] : colorMap) {
1125
0
        const QString line = image.format() != QImage::Format_RGB32 && !qAlpha(color)
1126
0
            ? QString::asprintf("\"%s c None\"", xpm_color_name(cpp, index))
1127
0
            : QString::asprintf("\"%s c #%02x%02x%02x\"", xpm_color_name(cpp, index),
1128
0
                                qRed(color), qGreen(color), qBlue(color));
1129
0
        s << ',' << Qt::endl << line;
1130
0
    }
1131
1132
    // write pixels, limit to 4 characters per pixel
1133
0
    for (int y = 0; y < h; ++y) {
1134
0
        s << ',' << Qt::endl << '\"';
1135
0
        const QRgb *yp = reinterpret_cast<const QRgb *>(image.constScanLine(y));
1136
0
        for (int x = 0; x < w; ++x)
1137
0
            s << xpm_color_name(cpp, colorMap[yp[x]]);
1138
0
        s << '\"';
1139
0
    }
1140
0
    s << "};" << Qt::endl;
1141
0
    return static_cast<bool>(s);
1142
0
}
1143
1144
QXpmHandler::QXpmHandler()
1145
0
    : state(Ready), index(0)
1146
0
{
1147
0
}
1148
1149
bool QXpmHandler::readHeader()
1150
0
{
1151
0
    state = Error;
1152
0
    if (!read_xpm_header(device(), nullptr, index, buffer, &cpp, &ncols, &width, &height))
1153
0
        return false;
1154
0
    state = ReadHeader;
1155
0
    return true;
1156
0
}
1157
1158
bool QXpmHandler::readImage(QImage *image)
1159
0
{
1160
0
    if (state == Error)
1161
0
        return false;
1162
1163
0
    if (state == Ready && !readHeader()) {
1164
0
        state = Error;
1165
0
        return false;
1166
0
    }
1167
1168
0
    if (!read_xpm_body(device(), nullptr, index, buffer, cpp, ncols, width, height, *image)) {
1169
0
        state = Error;
1170
0
        return false;
1171
0
    }
1172
1173
0
    state = Ready;
1174
0
    return true;
1175
0
}
1176
1177
bool QXpmHandler::canRead() const
1178
0
{
1179
0
    if (state == Ready && !canRead(device()))
1180
0
        return false;
1181
1182
0
    if (state != Error) {
1183
0
        setFormat("xpm");
1184
0
        return true;
1185
0
    }
1186
1187
0
    return false;
1188
0
}
1189
1190
bool QXpmHandler::canRead(QIODevice *device)
1191
0
{
1192
0
    if (!device) {
1193
0
        qCWarning(lcImageIo, "QXpmHandler::canRead() called with no device");
1194
0
        return false;
1195
0
    }
1196
1197
0
    char head[6];
1198
0
    if (device->peek(head, sizeof(head)) != sizeof(head))
1199
0
        return false;
1200
1201
0
    return qstrncmp(head, "/* XPM", 6) == 0;
1202
0
}
1203
1204
bool QXpmHandler::read(QImage *image)
1205
0
{
1206
0
    if (!canRead())
1207
0
        return false;
1208
0
    return readImage(image);
1209
0
}
1210
1211
bool QXpmHandler::write(const QImage &image)
1212
0
{
1213
0
    return write_xpm_image(image, device(), fileName);
1214
0
}
1215
1216
bool QXpmHandler::supportsOption(ImageOption option) const
1217
0
{
1218
0
    return option == Name
1219
0
        || option == Size
1220
0
        || option == ImageFormat;
1221
0
}
1222
1223
QVariant QXpmHandler::option(ImageOption option) const
1224
0
{
1225
0
    if (option == Name) {
1226
0
        return fileName;
1227
0
    } else if (option == Size) {
1228
0
        if (state == Error)
1229
0
            return QVariant();
1230
0
        if (state == Ready && !const_cast<QXpmHandler*>(this)->readHeader())
1231
0
            return QVariant();
1232
0
        return QSize(width, height);
1233
0
    } else if (option == ImageFormat) {
1234
0
        if (state == Error)
1235
0
            return QVariant();
1236
0
        if (state == Ready && !const_cast<QXpmHandler*>(this)->readHeader())
1237
0
            return QVariant();
1238
        // If we have more than 256 colors in the table, we need to
1239
        // figure out, if it contains transparency. That means reading
1240
        // the whole color table, which is too much work work pre-checking
1241
        // the image format
1242
0
        if (ncols <= 256)
1243
0
            return QImage::Format_Indexed8;
1244
0
        else
1245
0
            return QImage::Format_Invalid;
1246
0
    }
1247
1248
0
    return QVariant();
1249
0
}
1250
1251
void QXpmHandler::setOption(ImageOption option, const QVariant &value)
1252
0
{
1253
0
    if (option == Name)
1254
0
        fileName = value.toString();
1255
0
}
1256
1257
QT_END_NAMESPACE
1258
1259
#endif // QT_NO_IMAGEFORMAT_XPM