]> code.delx.au - refind/blob - libeg/load_bmp.c
Added documentation on refind-mkdefault script (bootcoup.html and
[refind] / libeg / load_bmp.c
1 /*
2 * libeg/load_bmp.c
3 * Loading function for BMP images
4 *
5 * Copyright (c) 2006 Christoph Pfisterer
6 * All rights reserved.
7 *
8 * Redistribution and use in source and binary forms, with or without
9 * modification, are permitted provided that the following conditions are
10 * met:
11 *
12 * * Redistributions of source code must retain the above copyright
13 * notice, this list of conditions and the following disclaimer.
14 *
15 * * Redistributions in binary form must reproduce the above copyright
16 * notice, this list of conditions and the following disclaimer in the
17 * documentation and/or other materials provided with the
18 * distribution.
19 *
20 * * Neither the name of Christoph Pfisterer nor the names of the
21 * contributors may be used to endorse or promote products derived
22 * from this software without specific prior written permission.
23 *
24 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
25 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
26 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
27 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
28 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
29 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
30 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
31 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
32 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
33 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
34 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
35 */
36
37 #include "libegint.h"
38
39 // BMP structures
40
41 #ifdef __MAKEWITH_GNUEFI
42 #pragma pack(1)
43
44 typedef struct {
45 UINT8 Blue;
46 UINT8 Green;
47 UINT8 Red;
48 UINT8 Reserved;
49 } BMP_COLOR_MAP;
50
51 typedef struct {
52 CHAR8 CharB;
53 CHAR8 CharM;
54 UINT32 Size;
55 UINT16 Reserved[2];
56 UINT32 ImageOffset;
57 UINT32 HeaderSize;
58 UINT32 PixelWidth;
59 UINT32 PixelHeight;
60 UINT16 Planes; // Must be 1
61 UINT16 BitPerPixel; // 1, 4, 8, or 24
62 UINT32 CompressionType;
63 UINT32 ImageSize; // Compressed image size in bytes
64 UINT32 XPixelsPerMeter;
65 UINT32 YPixelsPerMeter;
66 UINT32 NumberOfColors;
67 UINT32 ImportantColors;
68 } BMP_IMAGE_HEADER;
69
70 #pragma pack()
71 #endif
72
73 //
74 // Load BMP image
75 //
76
77 EG_IMAGE * egDecodeBMP(IN UINT8 *FileData, IN UINTN FileDataLength, IN UINTN IconSize, IN BOOLEAN WantAlpha)
78 {
79 EG_IMAGE *NewImage;
80 BMP_IMAGE_HEADER *BmpHeader;
81 BMP_COLOR_MAP *BmpColorMap;
82 UINTN x, y;
83 UINT8 *ImagePtr;
84 UINT8 *ImagePtrBase;
85 UINTN ImageLineOffset;
86 UINT8 ImageValue = 0, AlphaValue;
87 EG_PIXEL *PixelPtr;
88 UINTN Index, BitIndex;
89
90 // read and check header
91 if (FileDataLength < sizeof(BMP_IMAGE_HEADER) || FileData == NULL)
92 return NULL;
93 BmpHeader = (BMP_IMAGE_HEADER *) FileData;
94 if (BmpHeader->CharB != 'B' || BmpHeader->CharM != 'M')
95 return NULL;
96 if (BmpHeader->CompressionType != 0)
97 return NULL;
98 if (BmpHeader->BitPerPixel != 1 && BmpHeader->BitPerPixel != 4 &&
99 BmpHeader->BitPerPixel != 8 && BmpHeader->BitPerPixel != 24)
100 return NULL;
101
102 // calculate parameters
103 ImageLineOffset = BmpHeader->PixelWidth;
104 if (BmpHeader->BitPerPixel == 24)
105 ImageLineOffset *= 3;
106 else if (BmpHeader->BitPerPixel == 1)
107 ImageLineOffset = (ImageLineOffset + 7) >> 3;
108 else if (BmpHeader->BitPerPixel == 4)
109 ImageLineOffset = (ImageLineOffset + 1) >> 1;
110 if ((ImageLineOffset % 4) != 0)
111 ImageLineOffset = ImageLineOffset + (4 - (ImageLineOffset % 4));
112 // check bounds
113 if (BmpHeader->ImageOffset + ImageLineOffset * BmpHeader->PixelHeight > FileDataLength)
114 return NULL;
115
116 // allocate image structure and buffer
117 NewImage = egCreateImage(BmpHeader->PixelWidth, BmpHeader->PixelHeight, WantAlpha);
118 if (NewImage == NULL)
119 return NULL;
120 AlphaValue = WantAlpha ? 255 : 0;
121
122 // convert image
123 BmpColorMap = (BMP_COLOR_MAP *)(FileData + sizeof(BMP_IMAGE_HEADER));
124 ImagePtrBase = FileData + BmpHeader->ImageOffset;
125 for (y = 0; y < BmpHeader->PixelHeight; y++) {
126 ImagePtr = ImagePtrBase;
127 ImagePtrBase += ImageLineOffset;
128 PixelPtr = NewImage->PixelData + (BmpHeader->PixelHeight - 1 - y) * BmpHeader->PixelWidth;
129
130 switch (BmpHeader->BitPerPixel) {
131
132 case 1:
133 for (x = 0; x < BmpHeader->PixelWidth; x++) {
134 BitIndex = x & 0x07;
135 if (BitIndex == 0)
136 ImageValue = *ImagePtr++;
137
138 Index = (ImageValue >> (7 - BitIndex)) & 0x01;
139 PixelPtr->b = BmpColorMap[Index].Blue;
140 PixelPtr->g = BmpColorMap[Index].Green;
141 PixelPtr->r = BmpColorMap[Index].Red;
142 PixelPtr->a = AlphaValue;
143 PixelPtr++;
144 }
145 break;
146
147 case 4:
148 for (x = 0; x <= BmpHeader->PixelWidth - 2; x += 2) {
149 ImageValue = *ImagePtr++;
150
151 Index = ImageValue >> 4;
152 PixelPtr->b = BmpColorMap[Index].Blue;
153 PixelPtr->g = BmpColorMap[Index].Green;
154 PixelPtr->r = BmpColorMap[Index].Red;
155 PixelPtr->a = AlphaValue;
156 PixelPtr++;
157
158 Index = ImageValue & 0x0f;
159 PixelPtr->b = BmpColorMap[Index].Blue;
160 PixelPtr->g = BmpColorMap[Index].Green;
161 PixelPtr->r = BmpColorMap[Index].Red;
162 PixelPtr->a = AlphaValue;
163 PixelPtr++;
164 }
165 if (x < BmpHeader->PixelWidth) {
166 ImageValue = *ImagePtr++;
167
168 Index = ImageValue >> 4;
169 PixelPtr->b = BmpColorMap[Index].Blue;
170 PixelPtr->g = BmpColorMap[Index].Green;
171 PixelPtr->r = BmpColorMap[Index].Red;
172 PixelPtr->a = AlphaValue;
173 PixelPtr++;
174 }
175 break;
176
177 case 8:
178 for (x = 0; x < BmpHeader->PixelWidth; x++) {
179 Index = *ImagePtr++;
180 PixelPtr->b = BmpColorMap[Index].Blue;
181 PixelPtr->g = BmpColorMap[Index].Green;
182 PixelPtr->r = BmpColorMap[Index].Red;
183 PixelPtr->a = AlphaValue;
184 PixelPtr++;
185 }
186 break;
187
188 case 24:
189 for (x = 0; x < BmpHeader->PixelWidth; x++) {
190 PixelPtr->b = *ImagePtr++;
191 PixelPtr->g = *ImagePtr++;
192 PixelPtr->r = *ImagePtr++;
193 PixelPtr->a = AlphaValue;
194 PixelPtr++;
195 }
196 break;
197
198 }
199 }
200
201 return NewImage;
202 }
203
204 //
205 // Save BMP image
206 //
207
208 VOID egEncodeBMP(IN EG_IMAGE *Image, OUT UINT8 **FileDataReturn, OUT UINTN *FileDataLengthReturn)
209 {
210 BMP_IMAGE_HEADER *BmpHeader;
211 UINT8 *FileData;
212 UINTN FileDataLength;
213 UINT8 *ImagePtr;
214 UINT8 *ImagePtrBase;
215 UINTN ImageLineOffset;
216 EG_PIXEL *PixelPtr;
217 UINTN x, y;
218
219 ImageLineOffset = Image->Width * 3;
220 if ((ImageLineOffset % 4) != 0)
221 ImageLineOffset = ImageLineOffset + (4 - (ImageLineOffset % 4));
222
223 // allocate buffer for file data
224 FileDataLength = sizeof(BMP_IMAGE_HEADER) + Image->Height * ImageLineOffset;
225 FileData = AllocateZeroPool(FileDataLength);
226 if (FileData == NULL) {
227 Print(L"Error allocate %d bytes\n", FileDataLength);
228 *FileDataReturn = NULL;
229 *FileDataLengthReturn = 0;
230 return;
231 }
232
233 // fill header
234 BmpHeader = (BMP_IMAGE_HEADER *)FileData;
235 BmpHeader->CharB = 'B';
236 BmpHeader->CharM = 'M';
237 BmpHeader->Size = FileDataLength;
238 BmpHeader->ImageOffset = sizeof(BMP_IMAGE_HEADER);
239 BmpHeader->HeaderSize = 40;
240 BmpHeader->PixelWidth = Image->Width;
241 BmpHeader->PixelHeight = Image->Height;
242 BmpHeader->Planes = 1;
243 BmpHeader->BitPerPixel = 24;
244 BmpHeader->CompressionType = 0;
245 BmpHeader->XPixelsPerMeter = 0xb13;
246 BmpHeader->YPixelsPerMeter = 0xb13;
247
248 // fill pixel buffer
249 ImagePtrBase = FileData + BmpHeader->ImageOffset;
250 for (y = 0; y < Image->Height; y++) {
251 ImagePtr = ImagePtrBase;
252 ImagePtrBase += ImageLineOffset;
253 PixelPtr = Image->PixelData + (Image->Height - 1 - y) * Image->Width;
254
255 for (x = 0; x < Image->Width; x++) {
256 *ImagePtr++ = PixelPtr->b;
257 *ImagePtr++ = PixelPtr->g;
258 *ImagePtr++ = PixelPtr->r;
259 PixelPtr++;
260 }
261 }
262
263 *FileDataReturn = FileData;
264 *FileDataLengthReturn = FileDataLength;
265 }
266
267 /* EOF */