forked from ExternalVendorCode/Signal-Server
Large code cleanup of image rendering engine
This commit is contained in:
@@ -16,6 +16,7 @@ int ppm_init(PIMAGE_CTX ctx){
|
|||||||
|
|
||||||
buf_size = ctx->width * ctx->height * RGB_SIZE;
|
buf_size = ctx->width * ctx->height * RGB_SIZE;
|
||||||
|
|
||||||
|
/* Allocate the canvas buffer */
|
||||||
ctx->canvas = (uint8_t*) calloc(buf_size,sizeof(uint8_t));
|
ctx->canvas = (uint8_t*) calloc(buf_size,sizeof(uint8_t));
|
||||||
ctx->next_pixel = ctx->canvas;
|
ctx->next_pixel = ctx->canvas;
|
||||||
if(ctx->canvas == NULL)
|
if(ctx->canvas == NULL)
|
||||||
@@ -32,10 +33,7 @@ int ppm_add_pixel(PIMAGE_CTX ctx,const uint8_t r,const uint8_t g,const uint8_t b
|
|||||||
next[0] = r;
|
next[0] = r;
|
||||||
next[1] = g;
|
next[1] = g;
|
||||||
next[2] = b;
|
next[2] = b;
|
||||||
/*if(ctx->model == IMAGE_RGBA){
|
|
||||||
next[3] = a;
|
|
||||||
ctx->next_pixel += 1;
|
|
||||||
}*/
|
|
||||||
ctx->next_pixel += 3;
|
ctx->next_pixel += 3;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
196
image.cc
196
image.cc
@@ -1,3 +1,10 @@
|
|||||||
|
/*
|
||||||
|
* Generic image output handling. This feature allows
|
||||||
|
* for extensible image output formats and permits for
|
||||||
|
* cleaner image rendering code in the model generation
|
||||||
|
* routines by moving all of the file-format specific
|
||||||
|
* logic to be handled here.
|
||||||
|
*/
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
@@ -12,6 +19,9 @@ typedef int _add_pixel(PIMAGE_CTX,const uint8_t,const uint8_t,const uint8_t,cons
|
|||||||
typedef int _get_pixel(PIMAGE_CTX,const size_t,const size_t,const uint8_t*,const uint8_t*,const uint8_t*,const uint8_t*);
|
typedef int _get_pixel(PIMAGE_CTX,const size_t,const size_t,const uint8_t*,const uint8_t*,const uint8_t*,const uint8_t*);
|
||||||
typedef int _write(PIMAGE_CTX,FILE*);
|
typedef int _write(PIMAGE_CTX,FILE*);
|
||||||
|
|
||||||
|
int get_dt(struct image_dispatch_table *dt, IMAGE_FORMAT format);
|
||||||
|
int load_library(struct image_dispatch_table *dt);
|
||||||
|
|
||||||
struct image_dispatch_table{
|
struct image_dispatch_table{
|
||||||
_init *init;
|
_init *init;
|
||||||
_add_pixel *add_pixel;
|
_add_pixel *add_pixel;
|
||||||
@@ -19,64 +29,31 @@ struct image_dispatch_table{
|
|||||||
_write *write;
|
_write *write;
|
||||||
};
|
};
|
||||||
|
|
||||||
static IMAGE_FORMAT format = IMAGE_PPM;
|
static IMAGE_FORMAT default_format = IMAGE_PPM;
|
||||||
char *dynamic_backend = NULL;
|
char *dynamic_backend = NULL;
|
||||||
|
|
||||||
int load_library(struct image_dispatch_table *dt){
|
/*
|
||||||
void *hndl;
|
* image_set_format
|
||||||
int success = 0;
|
* Changes the default format for the next
|
||||||
|
* uninitialized image canvas
|
||||||
if(dynamic_backend == NULL){
|
*/
|
||||||
fprintf(stderr,"Custom image processor requested without specification\n");
|
int image_set_format(IMAGE_FORMAT format){
|
||||||
|
if(format <= IMAGE_DEFAULT || format >= IMAGE_FORMAT_MAX)
|
||||||
return EINVAL;
|
return EINVAL;
|
||||||
}
|
default_format = format;
|
||||||
|
return 0;
|
||||||
hndl = dlopen(dynamic_backend,RTLD_LAZY);
|
|
||||||
if(hndl == NULL){
|
|
||||||
fprintf(stderr,"Error loading shared object\n");
|
|
||||||
return EINVAL;
|
|
||||||
}
|
|
||||||
|
|
||||||
dt->init = (_init*)dlsym(hndl,"lib_init");
|
|
||||||
dt->add_pixel = (_add_pixel*)dlsym(hndl,"lib_add_pixel");
|
|
||||||
dt->get_pixel = (_get_pixel*)dlsym(hndl,"lib_get_pixel");
|
|
||||||
dt->write = (_write*)dlsym(hndl,"lib_write");
|
|
||||||
|
|
||||||
if(dt->init == NULL ||
|
|
||||||
dt->add_pixel == NULL ||
|
|
||||||
dt->get_pixel == NULL ||
|
|
||||||
dt->write == NULL){
|
|
||||||
fprintf(stderr,"Invalid image processing module specified\n");
|
|
||||||
success = dlclose(hndl);
|
|
||||||
}
|
|
||||||
|
|
||||||
return success;
|
|
||||||
}
|
|
||||||
|
|
||||||
int get_dt(struct image_dispatch_table *dt, IMAGE_FORMAT format){
|
|
||||||
int success = 0;
|
|
||||||
|
|
||||||
memset((void*)dt,0x00,sizeof(struct image_dispatch_table));
|
|
||||||
switch(format){
|
|
||||||
case IMAGE_PPM:
|
|
||||||
dt->init = ppm_init;
|
|
||||||
dt->add_pixel = ppm_add_pixel;
|
|
||||||
dt->get_pixel = ppm_get_pixel;
|
|
||||||
dt->write = ppm_write;
|
|
||||||
break;
|
|
||||||
case IMAGE_LIBRARY:
|
|
||||||
success = load_library(dt);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
success = EINVAL;
|
|
||||||
}
|
|
||||||
return success;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* image_init
|
||||||
|
* Initialize an image context. Must be called
|
||||||
|
* before attempting to write any image data
|
||||||
|
*/
|
||||||
int image_init(PIMAGE_CTX ctx, \
|
int image_init(PIMAGE_CTX ctx, \
|
||||||
const size_t width, \
|
const size_t width, \
|
||||||
const size_t height, \
|
const size_t height, \
|
||||||
const IMAGE_MODEL model) {
|
const IMAGE_MODEL model, \
|
||||||
|
const IMAGE_FORMAT format) {
|
||||||
|
|
||||||
int success = 0;
|
int success = 0;
|
||||||
struct image_dispatch_table *dt;
|
struct image_dispatch_table *dt;
|
||||||
@@ -88,7 +65,7 @@ int image_init(PIMAGE_CTX ctx, \
|
|||||||
return EINVAL;
|
return EINVAL;
|
||||||
if(model < 0 || model > IMAGE_MODEL_MAX)
|
if(model < 0 || model > IMAGE_MODEL_MAX)
|
||||||
return EINVAL;
|
return EINVAL;
|
||||||
if(format < 0 || format > IMAGE_FORMAT_MAX)
|
if(format >= IMAGE_FORMAT_MAX || (format == IMAGE_LIBRARY && dynamic_backend == NULL))
|
||||||
return EINVAL;
|
return EINVAL;
|
||||||
|
|
||||||
memset(ctx,0x00,sizeof(IMAGE_CTX));
|
memset(ctx,0x00,sizeof(IMAGE_CTX));
|
||||||
@@ -97,7 +74,10 @@ int image_init(PIMAGE_CTX ctx, \
|
|||||||
ctx->width = width;
|
ctx->width = width;
|
||||||
ctx->height = height;
|
ctx->height = height;
|
||||||
ctx->model = model;
|
ctx->model = model;
|
||||||
ctx->format = format;
|
if(format == IMAGE_DEFAULT)
|
||||||
|
ctx->format = default_format;
|
||||||
|
else
|
||||||
|
ctx->format = format;
|
||||||
|
|
||||||
/* Get the dispatch table for this image format */
|
/* Get the dispatch table for this image format */
|
||||||
dt = (struct image_dispatch_table*) calloc(1,sizeof(struct image_dispatch_table));
|
dt = (struct image_dispatch_table*) calloc(1,sizeof(struct image_dispatch_table));
|
||||||
@@ -105,7 +85,7 @@ int image_init(PIMAGE_CTX ctx, \
|
|||||||
fprintf(stderr,"Error allocating dispatch table\n");
|
fprintf(stderr,"Error allocating dispatch table\n");
|
||||||
return ENOMEM;
|
return ENOMEM;
|
||||||
}
|
}
|
||||||
success = get_dt(dt,format);
|
success = get_dt(dt,ctx->format);
|
||||||
if(success != 0){
|
if(success != 0){
|
||||||
fprintf(stderr,"Error locating dispatch table\n");
|
fprintf(stderr,"Error locating dispatch table\n");
|
||||||
free(dt);
|
free(dt);
|
||||||
@@ -125,12 +105,20 @@ int image_init(PIMAGE_CTX ctx, \
|
|||||||
|
|
||||||
return success;
|
return success;
|
||||||
}
|
}
|
||||||
|
/*
|
||||||
|
* image_add_pixel, image_set_pixel, image_get_pixel, image_write
|
||||||
|
* Various setters ad getters for assigning pixel data. image_write
|
||||||
|
* takes an open file handle and writes image data to it.
|
||||||
|
* These functions simply wrap the underlying format-specific functions
|
||||||
|
*/
|
||||||
int image_add_pixel(PIMAGE_CTX ctx, const uint8_t r, const uint8_t g, const uint8_t b, const uint8_t a){
|
int image_add_pixel(PIMAGE_CTX ctx, const uint8_t r, const uint8_t g, const uint8_t b, const uint8_t a){
|
||||||
struct image_dispatch_table *dt = (struct image_dispatch_table*)ctx->_dt;
|
struct image_dispatch_table *dt = (struct image_dispatch_table*)ctx->_dt;
|
||||||
|
if(ctx->initialized != 1) return EINVAL;
|
||||||
return dt->add_pixel(ctx,r,g,b,a);
|
return dt->add_pixel(ctx,r,g,b,a);
|
||||||
}
|
}
|
||||||
int image_set_pixel(PIMAGE_CTX ctx, const size_t x, const size_t y, const uint8_t r, const uint8_t g, const uint8_t b, const uint8_t a){
|
int image_set_pixel(PIMAGE_CTX ctx, const size_t x, const size_t y, const uint8_t r, const uint8_t g, const uint8_t b, const uint8_t a){
|
||||||
size_t block_size;
|
size_t block_size;
|
||||||
|
if(ctx->initialized != 1) return EINVAL;
|
||||||
block_size = ctx->model == IMAGE_RGB ? RGB_SIZE : RGBA_SIZE;
|
block_size = ctx->model == IMAGE_RGB ? RGB_SIZE : RGBA_SIZE;
|
||||||
ctx->next_pixel = ctx->canvas + (x * block_size) + (ctx->width * block_size * y);
|
ctx->next_pixel = ctx->canvas + (x * block_size) + (ctx->width * block_size * y);
|
||||||
struct image_dispatch_table *dt = (struct image_dispatch_table*)ctx->_dt;
|
struct image_dispatch_table *dt = (struct image_dispatch_table*)ctx->_dt;
|
||||||
@@ -138,23 +126,35 @@ int image_set_pixel(PIMAGE_CTX ctx, const size_t x, const size_t y, const uint8_
|
|||||||
}
|
}
|
||||||
int image_get_pixel(PIMAGE_CTX ctx, const size_t x, const size_t y, const uint8_t *r, const uint8_t *g, const uint8_t *b, const uint8_t *a){
|
int image_get_pixel(PIMAGE_CTX ctx, const size_t x, const size_t y, const uint8_t *r, const uint8_t *g, const uint8_t *b, const uint8_t *a){
|
||||||
struct image_dispatch_table *dt = (struct image_dispatch_table*)ctx->_dt;
|
struct image_dispatch_table *dt = (struct image_dispatch_table*)ctx->_dt;
|
||||||
|
if(ctx->initialized != 1) return EINVAL;
|
||||||
return dt->get_pixel(ctx,x,y,r,g,b,a);
|
return dt->get_pixel(ctx,x,y,r,g,b,a);
|
||||||
}
|
}
|
||||||
|
int image_write(PIMAGE_CTX ctx, FILE *fd){
|
||||||
|
struct image_dispatch_table *dt = (struct image_dispatch_table*)ctx->_dt;
|
||||||
|
if(ctx->initialized != 1) return EINVAL;
|
||||||
|
return dt->write(ctx,fd);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* image_get_filename
|
||||||
|
* Creates an appropriate file name using data supplied
|
||||||
|
* by the user. If the extension is already correct, return
|
||||||
|
* that; if not append if there is space
|
||||||
|
*/
|
||||||
int image_get_filename(PIMAGE_CTX ctx, char *out, size_t len_out, char *in){
|
int image_get_filename(PIMAGE_CTX ctx, char *out, size_t len_out, char *in){
|
||||||
size_t len_src;
|
size_t len_src;
|
||||||
size_t len_ext;
|
size_t len_ext;
|
||||||
int success = 0;
|
int success = 0;
|
||||||
|
|
||||||
if(ctx->initialized != 1){
|
if(ctx->initialized != 1)
|
||||||
fprintf(stderr,"Called get filename before initialization\n");
|
|
||||||
return EINVAL;
|
return EINVAL;
|
||||||
}
|
|
||||||
|
|
||||||
|
/* Get various lengths */
|
||||||
len_src = strlen(in);
|
len_src = strlen(in);
|
||||||
len_ext = strlen(ctx->extension);
|
len_ext = strlen(ctx->extension);
|
||||||
|
|
||||||
if(len_src == 0){
|
if(len_src == 0){
|
||||||
in = "output";
|
in = (char*)"output";
|
||||||
len_src = 6;
|
len_src = 6;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -178,10 +178,43 @@ int image_get_filename(PIMAGE_CTX ctx, char *out, size_t len_out, char *in){
|
|||||||
}
|
}
|
||||||
return success;
|
return success;
|
||||||
}
|
}
|
||||||
int image_write(PIMAGE_CTX ctx, FILE *fd){
|
|
||||||
struct image_dispatch_table *dt = (struct image_dispatch_table*)ctx->_dt;
|
/*
|
||||||
return dt->write(ctx,fd);
|
* get_dt
|
||||||
|
* Load the dispatch table for the specified image
|
||||||
|
* format. Currently only pixmap supported.
|
||||||
|
*/
|
||||||
|
int get_dt(struct image_dispatch_table *dt, IMAGE_FORMAT format){
|
||||||
|
int success = 0;
|
||||||
|
|
||||||
|
memset((void*)dt,0x00,sizeof(struct image_dispatch_table));
|
||||||
|
switch(format){
|
||||||
|
case IMAGE_PPM:
|
||||||
|
dt->init = ppm_init;
|
||||||
|
dt->add_pixel = ppm_add_pixel;
|
||||||
|
dt->get_pixel = ppm_get_pixel;
|
||||||
|
dt->write = ppm_write;
|
||||||
|
break;
|
||||||
|
case IMAGE_LIBRARY:
|
||||||
|
success = load_library(dt);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
success = EINVAL;
|
||||||
|
}
|
||||||
|
return success;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* ==WARNING==: Here be dragons!
|
||||||
|
* Experimantal features beyond this point.
|
||||||
|
* Only use if you are sure you know what you
|
||||||
|
* are doing!
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* image_set_library
|
||||||
|
* Set the library to use for generating images
|
||||||
|
*/
|
||||||
int image_set_library(char *library){
|
int image_set_library(char *library){
|
||||||
char *libname;
|
char *libname;
|
||||||
size_t length;
|
size_t length;
|
||||||
@@ -193,7 +226,48 @@ int image_set_library(char *library){
|
|||||||
strncpy(libname,library,length);
|
strncpy(libname,library,length);
|
||||||
|
|
||||||
dynamic_backend = libname;
|
dynamic_backend = libname;
|
||||||
format = IMAGE_LIBRARY;
|
default_format = IMAGE_LIBRARY;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
/*
|
||||||
|
* image_get_library
|
||||||
|
* Returns the current saved image rendering
|
||||||
|
* library
|
||||||
|
*/
|
||||||
|
char* image_get_library(){
|
||||||
|
return dynamic_backend;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* load_library
|
||||||
|
* External image processing: experimental feature
|
||||||
|
* Load an external library to perform image processing
|
||||||
|
* It must be a custom compatible library
|
||||||
|
*/
|
||||||
|
int load_library(struct image_dispatch_table *dt){
|
||||||
|
void *hndl;
|
||||||
|
int success = 0;
|
||||||
|
|
||||||
|
/* Validate object file */
|
||||||
|
if(dynamic_backend == NULL || strlen(dynamic_backend) == 0){
|
||||||
|
fprintf(stderr,"Custom image processor requested without specification\n");
|
||||||
|
return EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Load shared object and locate required exports */
|
||||||
|
hndl = dlopen(dynamic_backend,RTLD_LAZY);
|
||||||
|
if(hndl == NULL){
|
||||||
|
fprintf(stderr,"Error loading shared object\n");
|
||||||
|
return EINVAL;
|
||||||
|
}
|
||||||
|
/* Perform symbol lookup */
|
||||||
|
if((dt->init = (_init*)dlsym(hndl,"lib_init")) == NULL ||
|
||||||
|
(dt->add_pixel = (_add_pixel*)dlsym(hndl,"lib_add_pixel")) == NULL ||
|
||||||
|
(dt->get_pixel = (_get_pixel*)dlsym(hndl,"lib_get_pixel")) == NULL ||
|
||||||
|
(dt->write = (_write*)dlsym(hndl,"lib_write")) == NULL){
|
||||||
|
fprintf(stderr,"Invalid image processing module specified\n\t%s",dlerror());
|
||||||
|
success = dlclose(hndl);
|
||||||
|
}
|
||||||
|
|
||||||
|
return success;
|
||||||
|
}
|
20
image.hh
20
image.hh
@@ -6,7 +6,8 @@
|
|||||||
#define RGB_SIZE 3
|
#define RGB_SIZE 3
|
||||||
#define RGBA_SIZE 4
|
#define RGBA_SIZE 4
|
||||||
|
|
||||||
typedef enum _IMAGE_FORMAT{ IMAGE_PPM, \
|
typedef enum _IMAGE_FORMAT{ IMAGE_DEFAULT = 0, \
|
||||||
|
IMAGE_PPM, \
|
||||||
IMAGE_LIBRARY, \
|
IMAGE_LIBRARY, \
|
||||||
IMAGE_FORMAT_MAX \
|
IMAGE_FORMAT_MAX \
|
||||||
} IMAGE_FORMAT;
|
} IMAGE_FORMAT;
|
||||||
@@ -28,15 +29,16 @@ typedef struct _IMAGE_CTX{
|
|||||||
void *_dt;
|
void *_dt;
|
||||||
} IMAGE_CTX, *PIMAGE_CTX;
|
} IMAGE_CTX, *PIMAGE_CTX;
|
||||||
|
|
||||||
int image_init(PIMAGE_CTX ctx, const size_t width, const size_t height, const IMAGE_MODEL model);
|
int image_set_format(IMAGE_FORMAT);
|
||||||
int image_add_pixel(PIMAGE_CTX ctx, const uint8_t r, const uint8_t g, const uint8_t b, const uint8_t a);
|
int image_init(PIMAGE_CTX, const size_t, const size_t, const IMAGE_MODEL, const IMAGE_FORMAT);
|
||||||
int image_set_pixel(PIMAGE_CTX ctx, const size_t x, const size_t y, const uint8_t r, const uint8_t g, const uint8_t b, const uint8_t a);
|
int image_add_pixel(PIMAGE_CTX ctx, const uint8_t, const uint8_t, const uint8_t, const uint8_t);
|
||||||
int image_get_pixel(PIMAGE_CTX ctx,const size_t x,const size_t y, uint8_t const *r, uint8_t const *g, uint8_t const *b, uint8_t const *a);
|
int image_set_pixel(PIMAGE_CTX ctx, const size_t, const size_t, const uint8_t, const uint8_t, const uint8_t, const uint8_t);
|
||||||
int image_get_filename(PIMAGE_CTX ctx, char* out, size_t len, char* in);
|
int image_get_pixel(PIMAGE_CTX ctx,const size_t,const size_t, uint8_t const*, uint8_t const*, uint8_t const*, uint8_t const*);
|
||||||
int image_write(PIMAGE_CTX ctx, FILE *fd);
|
int image_get_filename(PIMAGE_CTX, char*, size_t, char*);
|
||||||
int image_set_library(char *library);
|
int image_write(PIMAGE_CTX, FILE*);
|
||||||
|
int image_set_library(char*);
|
||||||
|
|
||||||
#define ADD_PIXEL(ctx,r,g,b) image_add_pixel((ctx),(r),(g),(b),0)
|
#define ADD_PIXEL(ctx,r,g,b) image_add_pixel((ctx),(r),(g),(b),0xff)
|
||||||
#define ADD_PIXELA(ctx,r,g,b,a) image_add_pixel((ctx),(r),(g),(b),(a))
|
#define ADD_PIXELA(ctx,r,g,b,a) image_add_pixel((ctx),(r),(g),(b),(a))
|
||||||
|
|
||||||
#endif
|
#endif
|
@@ -533,7 +533,7 @@ void DoRxdPwr(char *filename, unsigned char geo, unsigned char kml,
|
|||||||
IMAGE_CTX ctx;
|
IMAGE_CTX ctx;
|
||||||
int success;
|
int success;
|
||||||
|
|
||||||
if((success = image_init(&ctx, width, (kml ? height : height + 30), IMAGE_RGB)) != 0){
|
if((success = image_init(&ctx, width, (kml ? height : height + 30), IMAGE_RGB, IMAGE_DEFAULT)) != 0){
|
||||||
fprintf(stderr,"Error initializing image\n");
|
fprintf(stderr,"Error initializing image\n");
|
||||||
exit(success);
|
exit(success);
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user