[sdf -> bsdf renderer] Hold a refrence to `FT_Memory'.

Due to the way the 8SED algorithm works we need to keep
a track of distances and nearest point of all the pixels/
grid points.

* src/sdf/ftbsdf.c (BSDF_TRaster): Added struct to
  hold the `FT_Memory' reference, to be used to allocate
  memory.

* src/sdf/ftbsdf.c (ED, BSDF_Worker): Added few more
  essential struct. `ED' contains infomation about the
  nearest point. `BSDF_Worker' is used to combine all
  the essentail parameters to functions.

* src/sdf/ftbsdf.c (bsdf_copy_source_to_target => 
  bsdf_init_distance_map): Renamed.

* src/sdf/ftbsdf.c (*): Various minor changes.

* src/sdf/sdf.c: Change the order of source include
  because `ftsdf.c' change some internal freetype
  defines which are required in `ftbsdf.c'.
This commit is contained in:
Anuj Verma 2020-07-25 10:55:09 +05:30 committed by anujverma
parent 99a4759290
commit 13180a3cd4
3 changed files with 141 additions and 44 deletions

View File

@ -1,3 +1,29 @@
2020-07-25 Anuj Verma <anujv@iitbhilai.ac.in>
[sdf -> bsdf renderer] Hold a refrence to `FT_Memory'.
Due to the way the 8SED algorithm works we need to keep
a track of distances and nearest point of all the pixels/
grid points.
* src/sdf/ftbsdf.c (BSDF_TRaster): Added struct to
hold the `FT_Memory' reference, to be used to allocate
memory.
* src/sdf/ftbsdf.c (ED, BSDF_Worker): Added few more
essential struct. `ED' contains infomation about the
nearest point. `BSDF_Worker' is used to combine all
the essentail parameters to functions.
* src/sdf/ftbsdf.c (bsdf_copy_source_to_target =>
bsdf_init_distance_map): Renamed.
* src/sdf/ftbsdf.c (*): Various minor changes.
* src/sdf/sdf.c: Change the order of source include
because `ftsdf.c' change some internal freetype
defines which are required in `ftbsdf.c'.
2020-07-25 Anuj Verma <anujv@iitbhilai.ac.in> 2020-07-25 Anuj Verma <anujv@iitbhilai.ac.in>
[sdf -> bsdf renderer] Convert 8bit bitmap to 16bit. [sdf -> bsdf renderer] Convert 8bit bitmap to 16bit.

View File

@ -1,6 +1,7 @@
#include <freetype/internal/ftobjs.h> #include <freetype/internal/ftobjs.h>
#include <freetype/internal/ftdebug.h> #include <freetype/internal/ftdebug.h>
#include <freetype/internal/ftmemory.h>
#include "ftsdf.h" #include "ftsdf.h"
#include "ftsdferrs.h" #include "ftsdferrs.h"
@ -26,7 +27,41 @@
* *
*/ */
typedef FT_Short FT_6D10; /* 6.10 fixed point representation */ typedef FT_Vector FT_16D16_Vec; /* with 16.16 fixed point components */
typedef FT_Short FT_6D10; /* 6.10 fixed point representation */
typedef FT_Fixed FT_16D16; /* 6.10 fixed point representation */
/**************************************************************************
*
* structs
*
*/
typedef struct BSDF_TRaster_
{
FT_Memory memory; /* used internally to allocate memory */
} BSDF_TRaster;
/* euclidean distance used for euclidean distance transform */
typedef struct ED_
{
FT_16D16 dist; /* distance at `near' */
FT_16D16_Vec near; /* nearest point */
} ED;
typedef struct BSDF_Worker_
{
ED* distance_map;
FT_Int width;
FT_Int rows;
SDF_Raster_Params params;
} BSDF_Worker;
/************************************************************************** /**************************************************************************
* *
@ -34,13 +69,24 @@
* *
*/ */
/* This function copy the `source' bitmap on top of */ /**************************************************************************
/* `target' bitmap's center. It also converts the */ *
/* values to 16bpp in a fixed point format. */ * @Function:
* bsdf_init_distance_map
*
* @Description:
* This function initialize the distance map according to
* algorithm '8-point sequential Euclidean distance mapping' (8SED).
*
* @Input:
* [TODO]
*
* @Return:
* [TODO]
*/
static FT_Error static FT_Error
bsdf_copy_source_to_target( const FT_Bitmap* source, bsdf_init_distance_map( const FT_Bitmap* source,
FT_Bitmap* target, BSDF_Worker* worker )
FT_Bool flip_y )
{ {
FT_Error error = FT_Err_Ok; FT_Error error = FT_Err_Ok;
FT_Bool is_monochrome = 0; FT_Bool is_monochrome = 0;
@ -49,10 +95,10 @@
FT_Int num_channels; FT_Int num_channels;
FT_Int t_i, t_j, s_i, s_j; FT_Int t_i, t_j, s_i, s_j;
FT_Byte* s; FT_Byte* s;
FT_6D10* t; ED* t;
/* again check the parameters (probably unnecessary) */ /* again check the parameters (probably unnecessary) */
if ( !source || !target ) if ( !source || !worker )
{ {
error = FT_THROW( Invalid_Argument ); error = FT_THROW( Invalid_Argument );
goto Exit; goto Exit;
@ -62,8 +108,8 @@
/* i.e. aligning the source to the center of the */ /* i.e. aligning the source to the center of the */
/* target, the target's width/rows must be checked */ /* target, the target's width/rows must be checked */
/* before copying. */ /* before copying. */
if ( target->width < source->width || if ( worker->width < source->width ||
target->rows < source->rows ) worker->rows < source->rows )
{ {
error = FT_THROW( Invalid_Argument ); error = FT_THROW( Invalid_Argument );
goto Exit; goto Exit;
@ -78,15 +124,6 @@
goto Exit; goto Exit;
} }
/* make sure target bitmap is 16bpp */
if ( target->pixel_mode != FT_PIXEL_MODE_GRAY16 )
{
FT_ERROR(( "[bsdf] bsdf_copy_source_to_target: "
"Invalid pixel mode of target bitmap" ));
error = FT_THROW( Invalid_Argument );
goto Exit;
}
#ifdef FT_DEBUG_LEVEL_TRACE #ifdef FT_DEBUG_LEVEL_TRACE
if ( source->pixel_mode == FT_PIXEL_MODE_MONO ) if ( source->pixel_mode == FT_PIXEL_MODE_MONO )
{ {
@ -100,13 +137,13 @@
/* Calculate the difference in width and rows */ /* Calculate the difference in width and rows */
/* of the target and source. */ /* of the target and source. */
x_diff = target->width - source->width; x_diff = worker->width - source->width;
y_diff = target->rows - source->rows; y_diff = worker->rows - source->rows;
x_diff /= 2; x_diff /= 2;
y_diff /= 2; y_diff /= 2;
t = (FT_6D10*)target->buffer; t = (ED*)worker->distance_map;
s = source->buffer; s = source->buffer;
/* For now we only support pixel mode `FT_PIXEL_MODE_MONO' */ /* For now we only support pixel mode `FT_PIXEL_MODE_MONO' */
@ -129,8 +166,8 @@
break; break;
case FT_PIXEL_MODE_GRAY: case FT_PIXEL_MODE_GRAY:
{ {
FT_Int t_width = target->width; FT_Int t_width = worker->width;
FT_Int t_rows = target->rows; FT_Int t_rows = worker->rows;
FT_Int s_width = source->width; FT_Int s_width = source->width;
FT_Int s_rows = source->rows; FT_Int s_rows = source->rows;
@ -149,15 +186,15 @@
s_i = t_i - x_diff; s_i = t_i - x_diff;
s_j = t_j - y_diff; s_j = t_j - y_diff;
/* assign 0 to the padding */ /* assign INT_MAX to the padding */
if ( s_i < 0 || s_i >= s_width || if ( s_i < 0 || s_i >= s_width ||
s_j < 0 || s_j >= s_rows ) s_j < 0 || s_j >= s_rows )
{ {
t[t_index] = 0; t[t_index].dist = FT_INT_MAX;
continue; continue;
} }
if ( flip_y ) if ( worker->params.flip_y )
s_index = ( s_rows - s_j - 1 ) * s_width + s_i; s_index = ( s_rows - s_j - 1 ) * s_width + s_i;
else else
s_index = s_j * s_width + s_i; s_index = s_j * s_width + s_i;
@ -169,12 +206,12 @@
if ( pixel_value == 255 ) if ( pixel_value == 255 )
pixel_value = 256; pixel_value = 256;
/* Assume that 256 is fractional value with */ /* Assume that 256 is fractional value with */
/* 0.8 representation, to make it 6.10 left */ /* 0.8 representation, to make it 16.16 left */
/* shift the value by 2. */ /* shift the value by 8. */
pixel_value <<= 2; pixel_value <<= 8;
t[t_index] = pixel_value; t[t_index].dist = pixel_value;
} }
} }
@ -202,10 +239,18 @@
bsdf_raster_new( FT_Memory memory, bsdf_raster_new( FT_Memory memory,
FT_Raster* araster) FT_Raster* araster)
{ {
FT_UNUSED( memory ); FT_Error error = FT_Err_Ok;
FT_UNUSED( araster ); BSDF_TRaster* raster = NULL;
return FT_Err_Ok;
*araster = 0;
if ( !FT_ALLOC( raster, sizeof( SDF_TRaster ) ) )
{
raster->memory = memory;
*araster = (FT_Raster)raster;
}
return error;
} }
static void static void
@ -236,9 +281,12 @@
bsdf_raster_render( FT_Raster raster, bsdf_raster_render( FT_Raster raster,
const FT_Raster_Params* params ) const FT_Raster_Params* params )
{ {
FT_Error error = FT_Err_Ok; FT_Error error = FT_Err_Ok;
FT_Bitmap* source = NULL; FT_Bitmap* source = NULL;
FT_Bitmap* target = NULL; FT_Bitmap* target = NULL;
FT_Memory memory = NULL;
BSDF_TRaster* bsdf_raster = (SDF_TRaster*)raster;
BSDF_Worker worker;
const SDF_Raster_Params* sdf_params = (const SDF_Raster_Params*)params; const SDF_Raster_Params* sdf_params = (const SDF_Raster_Params*)params;
@ -246,7 +294,7 @@
FT_UNUSED( raster ); FT_UNUSED( raster );
/* check for valid parameters */ /* check for valid parameters */
if ( !params ) if ( !raster || !params )
{ {
error = FT_THROW( Invalid_Argument ); error = FT_THROW( Invalid_Argument );
goto Exit; goto Exit;
@ -269,6 +317,16 @@
goto Exit; goto Exit;
} }
memory = bsdf_raster->memory;
if ( !memory )
{
FT_TRACE0(( "[bsdf] bsdf_raster_render:\n"
" Raster not setup properly, "
"unable to find memory handle.\n" ));
error = FT_THROW( Invalid_Handle );
goto Exit;
}
/* check if spread is set properly */ /* check if spread is set properly */
if ( sdf_params->spread > MAX_SPREAD || if ( sdf_params->spread > MAX_SPREAD ||
sdf_params->spread < MIN_SPREAD ) sdf_params->spread < MIN_SPREAD )
@ -285,8 +343,18 @@
goto Exit; goto Exit;
} }
FT_CALL( bsdf_copy_source_to_target( source, target, /* setup the worker */
sdf_params->flip_y ) );
/* allocate the distance map */
if ( !FT_QALLOC_MULT( worker.distance_map, target->rows,
target->width * sizeof( *worker.distance_map ) ) )
goto Exit;
worker.width = target->width;
worker.rows = target->rows;
worker.params = *sdf_params;
FT_CALL( bsdf_init_distance_map( source, &worker ) );
Exit: Exit:
return error; return error;
@ -295,7 +363,10 @@
static void static void
bsdf_raster_done( FT_Raster raster ) bsdf_raster_done( FT_Raster raster )
{ {
FT_UNUSED( raster ); FT_Memory memory = (FT_Memory)((SDF_TRaster*)raster)->memory;
FT_FREE( raster );
} }
FT_DEFINE_RASTER_FUNCS( FT_DEFINE_RASTER_FUNCS(

View File

@ -3,8 +3,8 @@
#define FT_MAKE_OPTION_SINGLE_OBJECT #define FT_MAKE_OPTION_SINGLE_OBJECT
#include "ftsdfrend.c" #include "ftsdfrend.c"
#include "ftsdf.c"
#include "ftbsdf.c" #include "ftbsdf.c"
#include "ftsdf.c"
/* END */ /* END */