Graphical demo for the LPC4088 Experiment Base Board with one of the Display Expansion Kits. This demo shows a rotating 3D cube with and without textures.
bmp.cpp@0:c828045bbe69, 2014-10-03 (annotated)
- Committer:
- embeddedartists
- Date:
- Fri Oct 03 12:48:37 2014 +0000
- Revision:
- 0:c828045bbe69
First version.
Who changed what in which revision?
User | Revision | Line number | New contents of line |
---|---|---|---|
embeddedartists | 0:c828045bbe69 | 1 | /* ---------------------------------------------------------------------------- |
embeddedartists | 0:c828045bbe69 | 2 | * ATMEL Microcontroller Software Support - ROUSSET - |
embeddedartists | 0:c828045bbe69 | 3 | * ---------------------------------------------------------------------------- |
embeddedartists | 0:c828045bbe69 | 4 | * Copyright (c) 2006, Atmel Corporation |
embeddedartists | 0:c828045bbe69 | 5 | |
embeddedartists | 0:c828045bbe69 | 6 | * All rights reserved. |
embeddedartists | 0:c828045bbe69 | 7 | * |
embeddedartists | 0:c828045bbe69 | 8 | * Redistribution and use in source and binary forms, with or without |
embeddedartists | 0:c828045bbe69 | 9 | * modification, are permitted provided that the following conditions are met: |
embeddedartists | 0:c828045bbe69 | 10 | * |
embeddedartists | 0:c828045bbe69 | 11 | * - Redistributions of source code must retain the above copyright notice, |
embeddedartists | 0:c828045bbe69 | 12 | * this list of conditions and the disclaiimer below. |
embeddedartists | 0:c828045bbe69 | 13 | * |
embeddedartists | 0:c828045bbe69 | 14 | * - Redistributions in binary form must reproduce the above copyright notice, |
embeddedartists | 0:c828045bbe69 | 15 | * this list of conditions and the disclaimer below in the documentation and/or |
embeddedartists | 0:c828045bbe69 | 16 | * other materials provided with the distribution. |
embeddedartists | 0:c828045bbe69 | 17 | * |
embeddedartists | 0:c828045bbe69 | 18 | * Atmel's name may not be used to endorse or promote products derived from |
embeddedartists | 0:c828045bbe69 | 19 | * this software without specific prior written permission. |
embeddedartists | 0:c828045bbe69 | 20 | * |
embeddedartists | 0:c828045bbe69 | 21 | * DISCLAIMER: THIS SOFTWARE IS PROVIDED BY ATMEL "AS IS" AND ANY EXPRESS OR |
embeddedartists | 0:c828045bbe69 | 22 | * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF |
embeddedartists | 0:c828045bbe69 | 23 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT ARE |
embeddedartists | 0:c828045bbe69 | 24 | * DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT, INDIRECT, |
embeddedartists | 0:c828045bbe69 | 25 | * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
embeddedartists | 0:c828045bbe69 | 26 | * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, |
embeddedartists | 0:c828045bbe69 | 27 | * OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF |
embeddedartists | 0:c828045bbe69 | 28 | * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING |
embeddedartists | 0:c828045bbe69 | 29 | * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, |
embeddedartists | 0:c828045bbe69 | 30 | * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
embeddedartists | 0:c828045bbe69 | 31 | * ---------------------------------------------------------------------------- |
embeddedartists | 0:c828045bbe69 | 32 | */ |
embeddedartists | 0:c828045bbe69 | 33 | |
embeddedartists | 0:c828045bbe69 | 34 | //------------------------------------------------------------------------------ |
embeddedartists | 0:c828045bbe69 | 35 | // Headers |
embeddedartists | 0:c828045bbe69 | 36 | //------------------------------------------------------------------------------ |
embeddedartists | 0:c828045bbe69 | 37 | |
embeddedartists | 0:c828045bbe69 | 38 | #include "mbed.h" |
embeddedartists | 0:c828045bbe69 | 39 | #include "bmp.h" |
embeddedartists | 0:c828045bbe69 | 40 | |
embeddedartists | 0:c828045bbe69 | 41 | //------------------------------------------------------------------------------ |
embeddedartists | 0:c828045bbe69 | 42 | // Internal constants |
embeddedartists | 0:c828045bbe69 | 43 | //------------------------------------------------------------------------------ |
embeddedartists | 0:c828045bbe69 | 44 | |
embeddedartists | 0:c828045bbe69 | 45 | /// BMP magic number ('BM'). |
embeddedartists | 0:c828045bbe69 | 46 | #define BMP_TYPE 0x4D42 |
embeddedartists | 0:c828045bbe69 | 47 | |
embeddedartists | 0:c828045bbe69 | 48 | //------------------------------------------------------------------------------ |
embeddedartists | 0:c828045bbe69 | 49 | // Internal types |
embeddedartists | 0:c828045bbe69 | 50 | //------------------------------------------------------------------------------ |
embeddedartists | 0:c828045bbe69 | 51 | |
embeddedartists | 0:c828045bbe69 | 52 | struct BMPPaletteEntry { |
embeddedartists | 0:c828045bbe69 | 53 | |
embeddedartists | 0:c828045bbe69 | 54 | unsigned char b; |
embeddedartists | 0:c828045bbe69 | 55 | unsigned char g; |
embeddedartists | 0:c828045bbe69 | 56 | unsigned char r; |
embeddedartists | 0:c828045bbe69 | 57 | unsigned char filler; |
embeddedartists | 0:c828045bbe69 | 58 | }; |
embeddedartists | 0:c828045bbe69 | 59 | |
embeddedartists | 0:c828045bbe69 | 60 | //------------------------------------------------------------------------------ |
embeddedartists | 0:c828045bbe69 | 61 | // Exported functions |
embeddedartists | 0:c828045bbe69 | 62 | //------------------------------------------------------------------------------ |
embeddedartists | 0:c828045bbe69 | 63 | |
embeddedartists | 0:c828045bbe69 | 64 | //------------------------------------------------------------------------------ |
embeddedartists | 0:c828045bbe69 | 65 | /// Returns 1 if the header of a BMP file is valid; otherwise returns 0. |
embeddedartists | 0:c828045bbe69 | 66 | /// \param file Buffer holding the file to examinate. |
embeddedartists | 0:c828045bbe69 | 67 | //------------------------------------------------------------------------------ |
embeddedartists | 0:c828045bbe69 | 68 | unsigned char BMP_IsValid(void *file) |
embeddedartists | 0:c828045bbe69 | 69 | { |
embeddedartists | 0:c828045bbe69 | 70 | return ((struct BMPHeader *) file)->type == BMP_TYPE; |
embeddedartists | 0:c828045bbe69 | 71 | } |
embeddedartists | 0:c828045bbe69 | 72 | |
embeddedartists | 0:c828045bbe69 | 73 | //------------------------------------------------------------------------------ |
embeddedartists | 0:c828045bbe69 | 74 | /// Returns the size of a BMP image given at least its header (the file does |
embeddedartists | 0:c828045bbe69 | 75 | /// not have to be complete). |
embeddedartists | 0:c828045bbe69 | 76 | /// \param file Pointer to the buffer which holds the BMP file. |
embeddedartists | 0:c828045bbe69 | 77 | //------------------------------------------------------------------------------ |
embeddedartists | 0:c828045bbe69 | 78 | unsigned int BMP_GetFileSize(void *file) |
embeddedartists | 0:c828045bbe69 | 79 | { |
embeddedartists | 0:c828045bbe69 | 80 | return ((struct BMPHeader *) file)->fileSize; |
embeddedartists | 0:c828045bbe69 | 81 | } |
embeddedartists | 0:c828045bbe69 | 82 | |
embeddedartists | 0:c828045bbe69 | 83 | //------------------------------------------------------------------------------ |
embeddedartists | 0:c828045bbe69 | 84 | /// Loads a BMP image located at the given address, decodes it and stores the |
embeddedartists | 0:c828045bbe69 | 85 | /// resulting image inside the provided buffer. Image must have the specified |
embeddedartists | 0:c828045bbe69 | 86 | /// width & height. |
embeddedartists | 0:c828045bbe69 | 87 | /// Returns 0 if the image has been loaded; otherwise returns an error code. |
embeddedartists | 0:c828045bbe69 | 88 | /// \param file Buffer which holds the BMP file. |
embeddedartists | 0:c828045bbe69 | 89 | /// \param buffer Buffer in which to store the decoded image. |
embeddedartists | 0:c828045bbe69 | 90 | /// \param width Buffer width in pixels. |
embeddedartists | 0:c828045bbe69 | 91 | /// \param height Buffer height in pixels. |
embeddedartists | 0:c828045bbe69 | 92 | /// \param bpp Number of bits per pixels that the buffer stores. |
embeddedartists | 0:c828045bbe69 | 93 | //------------------------------------------------------------------------------ |
embeddedartists | 0:c828045bbe69 | 94 | unsigned char BMP_Decode( |
embeddedartists | 0:c828045bbe69 | 95 | void *file, |
embeddedartists | 0:c828045bbe69 | 96 | unsigned char *buffer, |
embeddedartists | 0:c828045bbe69 | 97 | unsigned int width, |
embeddedartists | 0:c828045bbe69 | 98 | unsigned int height, |
embeddedartists | 0:c828045bbe69 | 99 | unsigned char bpp) |
embeddedartists | 0:c828045bbe69 | 100 | { |
embeddedartists | 0:c828045bbe69 | 101 | struct BMPHeader *header; |
embeddedartists | 0:c828045bbe69 | 102 | unsigned int i, j; |
embeddedartists | 0:c828045bbe69 | 103 | unsigned char r, g, b; |
embeddedartists | 0:c828045bbe69 | 104 | unsigned char *image; |
embeddedartists | 0:c828045bbe69 | 105 | |
embeddedartists | 0:c828045bbe69 | 106 | // Read header information |
embeddedartists | 0:c828045bbe69 | 107 | header = (struct BMPHeader *) file; |
embeddedartists | 0:c828045bbe69 | 108 | |
embeddedartists | 0:c828045bbe69 | 109 | // Verify that the file is valid |
embeddedartists | 0:c828045bbe69 | 110 | if (!BMP_IsValid(file)) { |
embeddedartists | 0:c828045bbe69 | 111 | |
embeddedartists | 0:c828045bbe69 | 112 | printf("BMP_Decode: File type is not 'BM' (0x%x).\n\r", |
embeddedartists | 0:c828045bbe69 | 113 | header->type); |
embeddedartists | 0:c828045bbe69 | 114 | return 1; |
embeddedartists | 0:c828045bbe69 | 115 | } |
embeddedartists | 0:c828045bbe69 | 116 | |
embeddedartists | 0:c828045bbe69 | 117 | // Check that parameters match |
embeddedartists | 0:c828045bbe69 | 118 | if ((header->compression != 0) |
embeddedartists | 0:c828045bbe69 | 119 | || (header->width != width) |
embeddedartists | 0:c828045bbe69 | 120 | || (header->height != height)) { |
embeddedartists | 0:c828045bbe69 | 121 | |
embeddedartists | 0:c828045bbe69 | 122 | printf("BMP_Decode: File format not supported\n\r"); |
embeddedartists | 0:c828045bbe69 | 123 | printf(" -> .compression = %d\n\r", header->compression); |
embeddedartists | 0:c828045bbe69 | 124 | printf(" -> .width = %d\n\r", header->width); |
embeddedartists | 0:c828045bbe69 | 125 | printf(" -> .height = %d\n\r", header->height); |
embeddedartists | 0:c828045bbe69 | 126 | printf(" -> .bits = %d\n\r", header->bits); |
embeddedartists | 0:c828045bbe69 | 127 | return 2; |
embeddedartists | 0:c828045bbe69 | 128 | } |
embeddedartists | 0:c828045bbe69 | 129 | |
embeddedartists | 0:c828045bbe69 | 130 | // Get image data |
embeddedartists | 0:c828045bbe69 | 131 | image = (unsigned char *) ((unsigned int) file + header->offset); |
embeddedartists | 0:c828045bbe69 | 132 | |
embeddedartists | 0:c828045bbe69 | 133 | // Check that the bpp resolution is supported |
embeddedartists | 0:c828045bbe69 | 134 | // Only a 24-bit output & 24- or 8-bit input are supported |
embeddedartists | 0:c828045bbe69 | 135 | if (bpp != 24) { |
embeddedartists | 0:c828045bbe69 | 136 | |
embeddedartists | 0:c828045bbe69 | 137 | printf("BMP_Decode: Output resolution not supported\n\r"); |
embeddedartists | 0:c828045bbe69 | 138 | return 3; |
embeddedartists | 0:c828045bbe69 | 139 | } |
embeddedartists | 0:c828045bbe69 | 140 | else if (header->bits == 24) { |
embeddedartists | 0:c828045bbe69 | 141 | |
embeddedartists | 0:c828045bbe69 | 142 | // Modified by Embedded Artists. This padding was not handled in the original |
embeddedartists | 0:c828045bbe69 | 143 | // implementation. Each row in the Pixel Array is padded to a multiple of 4 bytes in size |
embeddedartists | 0:c828045bbe69 | 144 | int rowWidthBytes = width * 3; |
embeddedartists | 0:c828045bbe69 | 145 | rowWidthBytes += (rowWidthBytes % 4); |
embeddedartists | 0:c828045bbe69 | 146 | |
embeddedartists | 0:c828045bbe69 | 147 | // Copy raw data from BMP to buffer (reversing row order) |
embeddedartists | 0:c828045bbe69 | 148 | for (i=0; i < height; i++) { |
embeddedartists | 0:c828045bbe69 | 149 | #if 1 |
embeddedartists | 0:c828045bbe69 | 150 | // Modified by Embedded Artists to convert to 565-format instead of 24 bit |
embeddedartists | 0:c828045bbe69 | 151 | int x; |
embeddedartists | 0:c828045bbe69 | 152 | for(x=0; x<width; x++) |
embeddedartists | 0:c828045bbe69 | 153 | { |
embeddedartists | 0:c828045bbe69 | 154 | r = *(unsigned char *)(image + (height - i - 1) * rowWidthBytes + 3*x + 2); |
embeddedartists | 0:c828045bbe69 | 155 | g = *(unsigned char *)(image + (height - i - 1) * rowWidthBytes + 3*x + 1); |
embeddedartists | 0:c828045bbe69 | 156 | b = *(unsigned char *)(image + (height - i - 1) * rowWidthBytes + 3*x + 0); |
embeddedartists | 0:c828045bbe69 | 157 | *(unsigned short *)(buffer + (i * width) * 2 + 2*x) = (((unsigned short)r & 0xF8) << 8) | |
embeddedartists | 0:c828045bbe69 | 158 | (((unsigned short)g & 0xFC) << 3) | |
embeddedartists | 0:c828045bbe69 | 159 | (((unsigned short)b & 0xF8) >> 3); |
embeddedartists | 0:c828045bbe69 | 160 | } |
embeddedartists | 0:c828045bbe69 | 161 | #else |
embeddedartists | 0:c828045bbe69 | 162 | memcpy(buffer + (i * width) * 3, |
embeddedartists | 0:c828045bbe69 | 163 | image + ((height - i - 1) * width) * 3, |
embeddedartists | 0:c828045bbe69 | 164 | width * 3); |
embeddedartists | 0:c828045bbe69 | 165 | #endif |
embeddedartists | 0:c828045bbe69 | 166 | } |
embeddedartists | 0:c828045bbe69 | 167 | |
embeddedartists | 0:c828045bbe69 | 168 | #if 0 |
embeddedartists | 0:c828045bbe69 | 169 | // Swap red and blue |
embeddedartists | 0:c828045bbe69 | 170 | for (i=0; i < height; i++) { |
embeddedartists | 0:c828045bbe69 | 171 | for (j=0; j < width; j++) { |
embeddedartists | 0:c828045bbe69 | 172 | |
embeddedartists | 0:c828045bbe69 | 173 | r = buffer[(i * width + j) * 3 + 2]; |
embeddedartists | 0:c828045bbe69 | 174 | g = buffer[(i * width + j) * 3 + 1]; |
embeddedartists | 0:c828045bbe69 | 175 | b = buffer[(i * width + j) * 3]; |
embeddedartists | 0:c828045bbe69 | 176 | |
embeddedartists | 0:c828045bbe69 | 177 | buffer[(i * width + j) * 3] = r; |
embeddedartists | 0:c828045bbe69 | 178 | buffer[(i * width + j) * 3 + 1] = g; |
embeddedartists | 0:c828045bbe69 | 179 | buffer[(i * width + j) * 3 + 2] = b; |
embeddedartists | 0:c828045bbe69 | 180 | } |
embeddedartists | 0:c828045bbe69 | 181 | } |
embeddedartists | 0:c828045bbe69 | 182 | #endif |
embeddedartists | 0:c828045bbe69 | 183 | } |
embeddedartists | 0:c828045bbe69 | 184 | else if (header->bits == 8) { |
embeddedartists | 0:c828045bbe69 | 185 | |
embeddedartists | 0:c828045bbe69 | 186 | // Retrieve palette |
embeddedartists | 0:c828045bbe69 | 187 | struct BMPPaletteEntry palette[256]; |
embeddedartists | 0:c828045bbe69 | 188 | memcpy(palette, |
embeddedartists | 0:c828045bbe69 | 189 | (unsigned char *) ((unsigned int) file + sizeof(struct BMPHeader)), |
embeddedartists | 0:c828045bbe69 | 190 | header->offset - sizeof(struct BMPHeader)); |
embeddedartists | 0:c828045bbe69 | 191 | |
embeddedartists | 0:c828045bbe69 | 192 | // Decode image (reversing row order) |
embeddedartists | 0:c828045bbe69 | 193 | for (i=0; i < height; i++) { |
embeddedartists | 0:c828045bbe69 | 194 | for (j=0; j < width; j++) { |
embeddedartists | 0:c828045bbe69 | 195 | |
embeddedartists | 0:c828045bbe69 | 196 | r = palette[image[(height - i - 1) * width + j]].r; |
embeddedartists | 0:c828045bbe69 | 197 | g = palette[image[(height - i - 1) * width + j]].g; |
embeddedartists | 0:c828045bbe69 | 198 | b = palette[image[(height - i - 1) * width + j]].b; |
embeddedartists | 0:c828045bbe69 | 199 | |
embeddedartists | 0:c828045bbe69 | 200 | buffer[(i * width + j) * 3] = r; |
embeddedartists | 0:c828045bbe69 | 201 | buffer[(i * width + j) * 3 + 1] = g; |
embeddedartists | 0:c828045bbe69 | 202 | buffer[(i * width + j) * 3 + 2] = b; |
embeddedartists | 0:c828045bbe69 | 203 | } |
embeddedartists | 0:c828045bbe69 | 204 | } |
embeddedartists | 0:c828045bbe69 | 205 | } |
embeddedartists | 0:c828045bbe69 | 206 | else { |
embeddedartists | 0:c828045bbe69 | 207 | |
embeddedartists | 0:c828045bbe69 | 208 | printf("BMP_Decode: Input resolution not supported\n\r"); |
embeddedartists | 0:c828045bbe69 | 209 | return 4; |
embeddedartists | 0:c828045bbe69 | 210 | } |
embeddedartists | 0:c828045bbe69 | 211 | |
embeddedartists | 0:c828045bbe69 | 212 | return 0; |
embeddedartists | 0:c828045bbe69 | 213 | } |
embeddedartists | 0:c828045bbe69 | 214 |