Save image data to sqlite

340 Views Asked by At

I have a function which loads an image from file and successfully creates an opengl texture from it.

/**
 * @brief Loads a texture from file and generates an OpenGL texture from it.
 * 
 * @param filename Path to the image file.
 * @param out_texture Texture id the results are bound to.
 * @param out_width Value pointer the resulting image width is written to.
 * @param out_height Value pointer the resulting image height is written to.
 * @param flip_image Stb indicator for flipping the image.
 * @return true Image has been successfully loaded.
 * @return false Failed loading the image.
 */
bool LoadTextureFromFile(const char *filename, GLuint *out_texture, int *out_width, int *out_height, bool flip_image = false)
{
    // Load from file
    int image_width = 0;
    int image_height = 0;
    stbi_set_flip_vertically_on_load(flip_image);
    unsigned char *image_data = stbi_load(filename, &image_width, &image_height, NULL, 4);
    
    if (image_data == NULL)
    {
        std::cout << "ERROR::Tools::GLHelper::LoadTextureFromFile  -   Failed to load image from file '" << filename << "'." << std::endl;

        stbi_image_free(image_data);
        return false;
    }

    // Create a OpenGL texture identifier
    GLuint image_texture;
    glGenTextures(1, &image_texture);
    glBindTexture(GL_TEXTURE_2D, image_texture);

    // Set texture wrapping parameters
    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
    // Set texture filtering parameters
    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);

    glPixelStorei(GL_UNPACK_ROW_LENGTH, 0);
    glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, image_width, image_height, 0, GL_RGBA, GL_UNSIGNED_BYTE, image_data);
    glGenerateMipmap(GL_TEXTURE_2D);

    *out_texture = image_texture;
    *out_width = image_width;
    *out_height = image_height;

    stbi_image_free(image_data);
    return true;
}

What I am trying to do is to load an image via stbi_load like above and save it as BLOB to sqlite. Afterwards I want to be able to load the very same blob and create an opengl texture from it in a separate function.

In the first step I created a function which only loads the image:

unsigned char *ImageDataFromFile(const char *filename)
{
    int image_width = 0;
    int image_height = 0;
    unsigned char *image_data = stbi_load(filename, &image_width, &image_height, NULL, 4);

    if (image_data == NULL)
    {
        std::cout << "ERROR::Tools::GLHelper::LoadTextureFromFile  -   Failed to load image from file '" << filename << "'." << std::endl;

        stbi_image_free(image_data);
    }

    return image_data;
}

In the next step I want to store this data into my sqlite database:

void DBConnector::AddImage(std::string name, unsigned char *data)
{
    sqlite3_stmt *stmt;
    int err = sqlite3_prepare_v2(db, "INSERT INTO images (name, img_data) VALUES (?, ?)", -1, &stmt, NULL);

    if (err != SQLITE_OK)
    {
        std::cout << "ERROR::DATA::DBConnector  -   Failed to prepare sqlite3 statement: \n"
                      << sqlite3_errmsg(db) << std::endl;
    }

    sqlite3_bind_text(stmt, 1, name.c_str(), -1, SQLITE_TRANSIENT);
    sqlite3_bind_blob(stmt, 2, data, -1, SQLITE_TRANSIENT);
    sqlite3_step(stmt);
    sqlite3_finalize(stmt);

    return;
}

Finally I connect the pieces:

unsigned char *image_data = Tools::FileHelper::ImageDataFromFile(selected_filepath.c_str());
db->AddImage("Foo", image_data);

What happens is that seemingly arbitrary data ends up in the database, which is definetly not image data. Sometimes the entries are just empty.

I suspect that I am handling the return type of stbi_load incorrectly, forcing random memory data into the database. Extract from stbi documentation:

The return value from an image loader is an 'unsigned char *' which points to the pixel data..

As I understand it I am simply passing the array pointer to sqlite3_bind_blob which accepts const void * just like glTexImage2D does. So why is it working for the one but not for the other? Or could the error source be somewhere else?

Edit

I also tried something else. Normally I pass -1 for size when calling i.e. sqlite3_bind_text, because the call will then automatically search for a null terminator. So I thought that I might have to pass the correct size in bytes when calling sqlite3_bind_blob because there might be no terminator there. So for an image with the size of 225 x 225 with 3 channels, I passed 225 * 225 * 3 as size parameter. Unfortunately this did not work either.

0

There are 0 best solutions below