1

I have written a primitive Bitmap Writer, that prepares a bitmap header, calculates a data section and write it as binary file.

The problem is, the program working for 800x600 resolution however, not working for some arbitrary resolution.

Bitmap Writer Code

#include <stdio.h>
#include <stdlib.h> // malloc()
#include <string.h>

/*
 * File Write , check this link : https://www.codingunit.com/c-tutorial-binary-file-io
 *
 */


typedef enum { FALSE, TRUE } boolean;

/* For 54 byte header data
 *
 * http://blog.paphus.com/blog/2012/08/14/write-your-own-bitmaps/
 */
 #pragma pack(push,1)
typedef struct BitmapHeaderType {
    unsigned char      b0;                      // 0
    unsigned char      b1;                      // 1
    unsigned int       fullFileSize;            // 2...5
    unsigned int       reserved;                // 6...9
    unsigned int       pixelOffset;             // 10..13 HEADER SIZE
    unsigned int       bitmapInfoHeaderSize;    // 14..17 BITMAP INFO HEADER SIZE
    unsigned int       pixelWidth;              // 18..21 image width
    unsigned int       pixelHeight;             // 22..25 image height
    unsigned short int numberOfColorPlanes;     // 26..27 the number of color planes
    unsigned short int bitPerPixel;             // 28..29 24bit, 32bit, bit size
    unsigned int       compessionState;         // 30..33 compression state, 0 for disable compression
    unsigned int       sizeOfRawPixel;          // 34..37 size of pixel data including paddings (24 bit padding changes data section)
    unsigned int       horizontalResolution;    // 38..41 just leave 2835
    unsigned int       verticalResolution;      // 42..45 just leave 2835
    unsigned int       numberOfColors;          // 46..49 set 0 for default
    unsigned int       numberOfImportantColors; // 50..53 set 0 for default
} BitmapHeader;
#pragma pack(pop)

void handleBitmapHeader(BitmapHeader *header, int width, int height);
int closestMultipleOfFour(int num);
void writeToFile(unsigned char* bytes, int len, char fileName[]);
void setPixel(unsigned char *data, int x, int y, unsigned char red, unsigned char green, unsigned char blue, int width, int height);

int main()
{
    //int width = 800; // compiling OK, runtime OK
    int width = 100; // compiling OK, runtime FAILED
    int height = 600;

    BitmapHeader *header  = (BitmapHeader *)malloc(sizeof(BitmapHeader));   

    handleBitmapHeader(header, width, height);

    unsigned char *data   = (unsigned char *)malloc(header->sizeOfRawPixel);
    unsigned char *bitmap = (unsigned char *)malloc(header->fullFileSize);

    // left top corner
    setPixel(data, 0        , 0         , 255,   0,   0, width, height);
    setPixel(data, 0        , 1         , 255,   0,   0, width, height);
    setPixel(data, 1        , 0         , 255,   0,   0, width, height);
    setPixel(data, 1        , 1         , 255,   0,   0, width, height);

    // right top corner
    setPixel(data, width-1  , 0         , 255,   0,   0, width, height);
    setPixel(data, width-1  , 1         , 255,   0,   0, width, height);
    setPixel(data, width-2  , 0         , 255,   0,   0, width, height);
    setPixel(data, width-2  , 1         , 255,   0,   0, width, height);

    // left bottom corner
    setPixel(data, 0        , height-1  , 255,   0,   0, width, height);
    setPixel(data, 0        , height-2  , 255,   0,   0, width, height);
    setPixel(data, 1        , height-1  , 255,   0,   0, width, height);
    setPixel(data, 1        , height-2  , 255,   0,   0, width, height);

    // right bottom corner
    setPixel(data, width-1  , height-1  , 255,   0,   0, width, height);
    setPixel(data, width-1  , height-2  , 255,   0,   0, width, height);
    setPixel(data, width-2  , height-1  , 255,   0,   0, width, height);
    setPixel(data, width-2  , height-2  , 255,   0,   0, width, height);

    // copy header to bitmap
    memcpy(bitmap, header, header->pixelOffset);

    // copy data to bitmap
    memcpy(bitmap+header->pixelOffset, data, header->fullFileSize);

    char fileName[] = "sampleBitmap.bmp";

    // write    
    writeToFile((unsigned char *)bitmap , header->fullFileSize, fileName);

    free(header);
    free(data);
    free(bitmap);

    return 0;
}

void handleBitmapHeader(BitmapHeader *header, int width, int height)
{
    // Calculate row with padding
    int rowWithPadding = closestMultipleOfFour(width*3);
    int rawSize = height * rowWithPadding;
    printf("Row With Padding : %4d\n", rowWithPadding);
    printf("Raw Size : Decimal[%4d], Hex[%4x]\n", rawSize, rawSize);

    header->b0 = 'B';
    header->b1 = 'M';
    header->fullFileSize = rawSize + 54;
    header->reserved = 0x00000000;
    header->pixelOffset = 54;
    header->bitmapInfoHeaderSize = 40;
    header->pixelWidth = (unsigned int)width;
    header->pixelHeight = (unsigned int)height;
    header->numberOfColorPlanes = 1;
    header->bitPerPixel = 24;
    header->compessionState = 0;
    header->sizeOfRawPixel = (unsigned int) rawSize;
    header->horizontalResolution = 0x2835;
    header->verticalResolution =   0x2835;
    header->numberOfColors = 0;
    header->numberOfImportantColors = 0;
}

int closestMultipleOfFour(int num)
{
    return (num + 3) & ~0x03;
}

void writeToFile(unsigned char* bytes, int len, char fileName[])
{

    FILE *filePtr;

    printf("HIT A : len = %d\n", len);

    filePtr = fopen(fileName, "wb");    // wb for write binary  

    printf("HIT B\n");

    if(!filePtr)
    {
        printf("ERROR::writeToFile()::Unable to open file : \"%s\"\n", fileName);
        return;
    }

    fwrite(bytes, len, 1, filePtr);

    fclose(filePtr);
}

void setPixel(unsigned char *data, int x, int y, unsigned char red, unsigned char green, unsigned char blue, int width, int height)
{
    y = height-1-y;

    int index = 0;
    int padSize = 0;

    if(x < width && y == 0)  // no need to calculate padding
        index = x * 3;
    else
    {
        boolean isPadding = ( (width*3) % 4 == 0) ? FALSE : TRUE;

        if(isPadding == TRUE) {
            padSize = closestMultipleOfFour(width * 3);

        } else {
            index = (y*width + x)*3;
        }
    }

    int bytes = 0;
    *(data + index+padSize + bytes++) = blue;
    *(data + index+padSize + bytes++) = green;
    *(data + index+padSize + bytes)   = red;

}

Please notify the lines below;

//int width = 800; // compiling OK, runtime OK
int width = 100; // compiling OK, runtime FAILED
int height = 600;

If you just run for 800x600 resolution like;

int width = 800; // compiling OK, runtime OK
/int width = 100; // compiling OK, runtime FAILED
int height = 600;

The program runs sucessfully. And also notify that for both conditions, the program is compiling successfully.

I try with gcc and MS VS, the behaviour is the same for both resolutions.

I've also tried to debug the program with MS VS but while debugging, the program triggered a breakpoint then directly ask me for a dll (I'm not expert on MS VS Debugging with C)

It seems the problematic part is with this line where exist in writeToFile() function;

filePtr = fopen(fileName, "wb");    // wb for write binary  

Any ideas on where do I've gone wrong ?

1 Answer 1

3

The program crashed because this line is accessing the memory out of bound:

memcpy(bitmap + header->pixelOffset, data, header->fullFileSize);

I think what you need is

memcpy(bitmap + header->pixelOffset, data, header->sizeOfRawPixel);

because bitmap is a buffer contains the header and the raw data.

Sign up to request clarification or add additional context in comments.

1 Comment

Exactly, thank you very much, I've written this so fast thus I've mistakenly assign full file size instead of data section but your answer save the day dude :)

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.