nuttx-apps/graphics/traveler/tools/libwld/wld_rgblookup.c

330 lines
10 KiB
C
Raw Normal View History

2016-11-10 18:42:37 +01:00
/****************************************************************************
* apps/graphics/traveler/tools/libwld/wld_rgblookup.c
*
* Copyright (C) 2016 Gregory Nutt. All rights reserved.
* Author: Gregory Nutt <gnutt@nuttx.org>
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
* 3. Neither the name NuttX nor the names of its contributors may be
* used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*
****************************************************************************/
/****************************************************************************
* Included Files
****************************************************************************/
#include <stdlib.h>
#include <math.h>
#include "trv_types.h"
#include "wld_mem.h"
#include "wld_utils.h"
#include "wld_color.h"
/****************************************************************************
* Pre-processor Definitions
****************************************************************************/
#undef MIN
#undef MAX
#define MIN(a,b) ((a) < (b) ? (a) : (b))
#define MAX(a,b) ((a) > (b) ? (a) : (b))
/****************************************************************************
* Private Data
****************************************************************************/
#if RGB_CUBE_SIZE < MIN_LUM_LEVELS
/* These arrays map color forms into g_unit_vector array indices */
static const enum unit_vector_index_e g_wld_bgrform_map[NCOLOR_FORMS] =
{
BLUE_NDX, GREENERBLUE_NDX, BLUEGREEN_NDX, LIGHTBLUE_NDX, GREY_NDX
};
static const enum unit_vector_index_e g_wld_brgform_map[NCOLOR_FORMS] =
{
BLUE_NDX, BLUEVIOLET_NDX, VIOLET_NDX, LIGHTBLUE_NDX, GREY_NDX
};
static const enum unit_vector_index_e g_wld_gbrform_map[NCOLOR_FORMS] =
{
GREEN_NDX, BLUERGREN_NDX, BLUEGREEN_NDX, LIGHTGREEN_NDX, GREY_NDX
};
static const enum unit_vector_index_e g_wld_grbform_map[NCOLOR_FORMS] =
{
GREEN_NDX, YELLOWGREEN_NDX, YELLOW_NDX, LIGHTGREEN_NDX, GREY_NDX
};
static const enum unit_vector_index_e g_wld_rbgform_map[NCOLOR_FORMS] =
{
RED_NDX, REDVIOLET_NDX, VIOLET_NDX, PINK_NDX, GREY_NDX
};
static const enum unit_vector_index_e g_wld_rgbform_map[NCOLOR_FORMS] =
{
RED_NDX, ORANGE_NDX, YELLOW_NDX, PINK_NDX, GREY_NDX
};
#else
static color_rgb_t *g_devpixel_lut = NULL;
static float g_wld_cube2pixel;
#endif
/****************************************************************************
* Public Data
****************************************************************************/
#if RGB_CUBE_SIZE < MIN_LUM_LEVELS
FAR color_lum_t *g_pixel2um_lut;
/* The following defines the "form" of each color in the g_unit_vector array */
const color_form_t g_wld_colorform[NCOLOR_FORMS] =
{
{ 1.0, 0.0, 0.0 },
{ 0.875, 0.4841229, 0.0 },
{ 0.7071068, 0.7071068, 0.0 },
{ 0.6666667, 0.5270463, 0.5270463 },
{ 0.5773503, 0.5773503, 0.5773503 }
};
/* This array defines each color supported in the luminance model */
const color_lum_t g_unit_vector[NUNIT_VECTORS] =
{
{ 0.5773503, 0.5773503, 0.5773503, 441.672932,}, /* GREY_NDX */
{ 0.0, 0.0, 1.0, 255.0 }, /* BLUE_NDX */
{ 0.0, 0.4841229, 0.875, 291.428571 }, /* GREENERBLUE_NDX */
{ 0.0, 0.7071068, 0.7071068, 360.624445 }, /* BLUEGREEN_NDX */
{ 0.4841229, 0.0, 0.875, 291.428571 }, /* BLUEVIOLET_NDX */
{ 0.7071068, 0.0, 0.7071068, 360.624445 }, /* VIOLET_NDX */
{ 0.5270463, 0.5270463, 0.6666667, 382.499981 }, /* LIGHTBLUE_NDX */
{ 0.0, 1.0, 0.0, 255.0 }, /* GREEN_NDX */
{ 0.0, 0.875, 0.4841229, 291.428571 }, /* BLUERGREN_NDX */
{ 0.4841229, 0.875, 0.0, 291.428571 }, /* YELLOWGREEN_NDX */
{ 0.7071068, 0.7071068, 0.0, 360.624445 }, /* YELLOW_NDX */
{ 0.5270463, 0.6666667, 0.5270463, 382.499981 }, /* LIGHTGREEN_NDX */
{ 1.0, 0.0, 0.0, 255.0 }, /* RED_NDX */
{ 0.875, 0.0, 0.4841229, 291.428571 }, /* REDVIOLET_NDX */
{ 0.875, 0.4841229, 0.0, 291.428571 }, /* ORANGE_NDX */
{ 0.6666667, 0.5270463, 0.5270463, 382.499981 }, /* PINK_NDX */
};
#endif
/****************************************************************************
* Public Functions
****************************************************************************/
/****************************************************************************
* Name: wld_rgblookup_allocate
*
* Description:
****************************************************************************/
void wld_rgblookup_allocate(void)
{
#if RGB_CUBE_SIZE < MIN_LUM_LEVELS
dev_pixel_t *lut;
int uvndx;
int lumndx;
int index;
/* Check if a color lookup table has been allocated */
g_devpixel_lut = (dev_pixel_t*)
wld_malloc(sizeof(dev_pixel_t) * (NUNIT_VECTORS*NLUMINANCES));
if (!g_devpixel_lut)
{
wld_fatal_error("ERROR: Failed to allocate color lookup table\n");
}
lut = g_devpixel_lut;
/* Save the color information and color lookup table for use in
* color mapping below.
*/
g_pixel2um_lut = (color_lum_t*)
wld_malloc(sizeof(color_lum_t) * (NUNIT_VECTORS*NLUMINANCES));
if (!g_pixel2um_lut)
{
wld_fatal_error("ERROR: Failed to allocate luminance table\n");
}
/* Allocate each color at each luminance value */
index = 0;
for (uvndx = 0; uvndx < NUNIT_VECTORS; uvndx++)
{
for (lumndx = 0; lumndx < NLUMINANCES; lumndx++)
{
color_rgb_t color;
FAR color_lum_t *lum;
/* Get a convenience pointer to the lookup table entry */
lum = &g_pixel2um_lut[index];
*lum = g_unit_vector[uvndx];
/* Get the luminance associated with this lum for this
* unit vector.
*/
lum->luminance = (lum->luminance * (float)(lumndx + 1)) / NLUMINANCES;
/* Convert to RGB and allocate the color */
color.red = (short) (lum->red * lum->luminance);
color.green = (short) (lum->green * lum->luminance);
color.blue = (short) (lum->blue * lum->luminance);
/* Save the RGB to pixel lookup data */
lut[index] = WLD_MKRGB(color.red, color.green, color.blue);
}
}
#else
dev_pixel_t *lut;
int index;
color_rgb_t rgb;
/* Check if a color lookup table has been allocated */
g_devpixel_lut = (dev_pixel_t*)
wld_malloc(sizeof(dev_pixel_t) * (WLD_PIXEL_MAX+1));
if (!g_devpixel_lut)
{
wld_fatal_error("ERROR: Failed to allocate color lookup table\n");
}
/* Save the color information and color lookup table for use in
* subsequent color mapping.
*/
lut = g_devpixel_lut;
/* Check if a Pixel-to-RGB color mapping table has been allocated */
g_devpixel_lut = (color_rgb_t*)
wld_malloc(sizeof(color_rgb_t) * (WLD_PIXEL_MAX+1));
if (!g_devpixel_lut)
{
wld_fatal_error("ERROR: Failed to allocate luminance table\n");
}
for (index = 0; index <= WLD_PIXEL_MAX; index++)
{
g_devpixel_lut[index].red
= g_devpixel_lut[index].green
= g_devpixel_lut[index].blue = 0;
}
/* Calculate the cube to trv_pixel_t scale factor. This factor will
* convert an RGB component in the range {0..RGB_CUBE_SIZE-1} to
* a value in the range {0..WLD_PIXEL_MAX}.
*/
g_wld_cube2pixel = (float)WLD_PIXEL_MAX / (float)(RGB_CUBE_SIZE-1);
/* Allocate each color in the RGB Cube */
for (rgb.red = 0; rgb.red < RGB_CUBE_SIZE; rgb.red++)
for (rgb.green = 0; rgb.green < RGB_CUBE_SIZE; rgb.green++)
for (rgb.blue = 0; rgb.blue < RGB_CUBE_SIZE; rgb.blue++)
{
color_rgb_t color;
color.red = (short) (rgb.red * 65535 / (RGB_CUBE_SIZE - 1));
color.green = (short) (rgb.green * 65535 / (RGB_CUBE_SIZE - 1));
color.blue = (short) (rgb.blue * 65535 / (RGB_CUBE_SIZE - 1));
/* Save the RGB to pixel lookup data */
lut[index] = WLD_MKRGB(color.red, color.green, color.blue);
/* Save the pixel to RGB lookup data */
if (color.pixel <= WLD_PIXEL_MAX)
{
g_devpixel_lut[color.pixel] = rgb;
}
}
#endif
}
/****************************************************************************
* Name: wld_color_endmapping
*
* Description:
* When all color mapping has been performed, this function should be
* called to release all resources dedicated to color mapping.
*
****************************************************************************/
void wld_color_endmapping(void)
{
#if RGB_CUBE_SIZE < MIN_LUM_LEVELS
if (g_pixel2um_lut)
{
wld_free(g_pixel2um_lut);
g_pixel2um_lut = NULL;
}
#else
if (g_devpixel_lut)
{
wld_free(g_devpixel_lut);
g_devpixel_lut = NULL;
}
#endif
}
/****************************************************************************
* Name: wld_rgblookup_free
*
* Description:
* Free the color lookup table
*
****************************************************************************/
void wld_rgblookup_free(void)
{
if (g_devpixel_lut)
{
wld_free(g_devpixel_lut);
g_devpixel_lut = NULL;
}
}