Goal
I'm trying to move towards Boost GIL to replace some similar functionality I've implemented that is reaching the end of its maintainable life.
I have existing code that works with 24 BPP, 8bit RGB images using uint8_t*
. I can't change that as the same interface is used to expose images from different places (e.g. OpenGL buffers) and there's already quite a lot of code.
Therefore I'm trying to use GIL in small steps, starting by reading a file and copying the pixels byte by byte into a std::vector<uint8_t>
which I can use to manage the storage, but still get a uint8_t*
by using &vector[0]
.
This can be dropped in transparently behind the existing interfaces until it's at a point where refactoring makes sense.
What I've tried
I thought this should be a simple case of using copy_pixels()
with two appropriately constructed views.
I've put together a minimal, complete example that illustrates the sum of what I've manage to achieve by looking over the documents and trying things out:
#include <boost/gil/rgb.hpp>
#include <boost/gil/extension/io/png_dynamic_io.hpp>
#include <stdint.h>
#include <vector>
int main() {
std::vector<uint8_t> storage;
{
using namespace boost::gil;
rgb8_image_t img;
png_read_image("test.png", img);
// what should replace 3 here to be more "generic"?
storage.resize(img.width()*img.height()*3);
// doesn't work, the type of the images aren't compatible.
copy_pixels(const_view(img),
interleaved_view(img.width(), img.height(), &storage[0], 3*img.width()));
}
}
Where I'm stuck
This doesn't compile:
error: cannot convert ‘const boost::gil::pixel<unsigned char, boost::gil::layout<boost::mpl::vector3<boost::gil::red_t, boost::gil::green_t, boost::gil::blue_t> > >’ to ‘unsigned char’ in assignment
Which is fairly self-explanatory - an RGB pixel can't be converted to a single unsigned char
automatically. I thought I'd try using copy_and_convert_pixels()
to fix this, but I can't see a way around the 3:1 (i.e. I have 3 unsigned char
s in the output for every pixel in the source image) problem with these conversions. Conversion seems to be more aimed at colourspace conversions (e.g. RGB->HSV) or packing changes.
A complete, simplified form of the actual code I ended up using is:
I originally had the following before I included any of the GIL headers:
I'm not sure exactly what issue they fixed with the version of boost I had at the time, but they don't seem to be required with 1.48.
Here's some code I once used:
It's only for grayscale, but presumably the coloured version is similar.
I would just push_back each color of the rgb8_pixel_t individually:
...but I'm not an expert on GIL either.
I just ran into this same problem; here for possible future reference is the answer I can come up with now that I've solved it for myself:
That copy_pixels approach is good. The only problem is the destination type. If you happen to know that rgb8_pixel_t is formatted in memory as if it were three consecutive uint8_t's, then all you need to do is something like this:
That's copied from my project, more or less; I'm using a 32-bit image but it should work the same for any other single, hardcoded image type. (I haven't learned how to use the "any_" stuff from GIL, so I can't comment on dynamically determined image types.)
In the code above, I somewhat crudely confirm that an rgba8_pixel_t is what OpenGL will see as an "INT_8_8_8_8" or whatever, by doing that static_assert. I think it's probably better to take that information from the GIL documentation than to make a guess and try to confirm it with an assertion, but I can't seem to find a clear statement of it there (I'm also new to GIL, so maybe I'm just missing it). But it seems fairly clear that this is part of the intention of the design of GIL's pixel types. For instance, the GIL design guide says at one point, "The most commonly used pixel is a homogeneous pixel whose values are together in memory." "Together in memory" seems to be just what I'm looking for. Just after that, the guide talks about "planar" pixel types, in which the color channel values for one pixel are NOT stored together in memory. It would be strange to go to all the trouble of supporting that distinction as carefully as they do, and then not actually bother to make the interleaved pixel type pack its color values tightly together in memory.
Anyway, I've demonstrated in my own project that this approach works with at least the version of Boost that I'm using (1.57), and I claim that if a future version changes this, then my static_assert will almost certainly catch it.
(Another approach to potentially fall back on would be to actually go ahead and use a planar pixel to map between your uint_8_t array and the rgb8_pixel_t that for_each_pixel gives you:
But this isn't really superior to the at_c strategy. It's just kind of a neat example I guess. *_planar_ptr_t is surprisingly smart!)
Also note that in today's C++ you don't have to make a separate type to capture the body of your "for each" loop; you can use an anonymous function, as above. (I wrap mine in std::function because I guess GIL does some internal copy-assignment of the function object, or something like that, and the compiler gets mad if a naked anonymous function is passed in. I guess the std::function wrapping may reduce efficiency a bit here; in my case that doesn't seem to be important.)