170 lines
4.8 KiB
C
170 lines
4.8 KiB
C
/*
|
|
* Copyright (C) 2016 Glyptodon, Inc.
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
* in the Software without restriction, including without limitation the rights
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
* THE SOFTWARE.
|
|
*/
|
|
|
|
#include "config.h"
|
|
#include "image-stream.h"
|
|
#include "jpeg.h"
|
|
#include "log.h"
|
|
#include "png.h"
|
|
|
|
#ifdef ENABLE_WEBP
|
|
#include "webp.h"
|
|
#endif
|
|
|
|
#include <cairo/cairo.h>
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
guacenc_decoder_mapping guacenc_decoder_map[] = {
|
|
{"image/png", guacenc_png_decoder},
|
|
{"image/jpeg", guacenc_jpeg_decoder},
|
|
#ifdef ENABLE_WEBP
|
|
{"image/webp", guacenc_webp_decoder},
|
|
#endif
|
|
{NULL, NULL}
|
|
};
|
|
|
|
guacenc_decoder* guacenc_get_decoder(const char* mimetype) {
|
|
|
|
/* Search through mapping for the decoder having given mimetype */
|
|
guacenc_decoder_mapping* current = guacenc_decoder_map;
|
|
while (current->mimetype != NULL) {
|
|
|
|
/* Return decoder if mimetype matches */
|
|
if (strcmp(current->mimetype, mimetype) == 0)
|
|
return current->decoder;
|
|
|
|
/* Next candidate decoder */
|
|
current++;
|
|
|
|
}
|
|
|
|
/* No such decoder */
|
|
guacenc_log(GUAC_LOG_WARNING, "Support for \"%s\" not present", mimetype);
|
|
return NULL;
|
|
|
|
}
|
|
|
|
guacenc_image_stream* guacenc_image_stream_alloc(int mask, int index,
|
|
const char* mimetype, int x, int y) {
|
|
|
|
/* Allocate stream */
|
|
guacenc_image_stream* stream = malloc(sizeof(guacenc_image_stream));
|
|
if (stream == NULL)
|
|
return NULL;
|
|
|
|
/* Init properties */
|
|
stream->index = index;
|
|
stream->mask = mask;
|
|
stream->x = x;
|
|
stream->y = y;
|
|
|
|
/* Associate with corresponding decoder */
|
|
stream->decoder = guacenc_get_decoder(mimetype);
|
|
|
|
/* Allocate initial buffer */
|
|
stream->length = 0;
|
|
stream->max_length = GUACENC_IMAGE_STREAM_INITIAL_LENGTH;
|
|
stream->buffer = (unsigned char*) malloc(stream->max_length);
|
|
|
|
return stream;
|
|
|
|
}
|
|
|
|
int guacenc_image_stream_receive(guacenc_image_stream* stream,
|
|
unsigned char* data, int length) {
|
|
|
|
/* Allocate more space if necessary */
|
|
if (stream->max_length - stream->length < length) {
|
|
|
|
/* Calculate a reasonable new max length guaranteed to fit buffer */
|
|
int new_max_length = stream->max_length * 2 + length;
|
|
|
|
/* Attempt to resize buffer */
|
|
unsigned char* new_buffer =
|
|
(unsigned char*) realloc(stream->buffer, new_max_length);
|
|
if (new_buffer == NULL)
|
|
return 1;
|
|
|
|
/* Store updated buffer and size */
|
|
stream->buffer = new_buffer;
|
|
stream->max_length = new_max_length;
|
|
|
|
}
|
|
|
|
/* Append data */
|
|
memcpy(stream->buffer + stream->length, data, length);
|
|
stream->length += length;
|
|
return 0;
|
|
|
|
}
|
|
|
|
int guacenc_image_stream_end(guacenc_image_stream* stream,
|
|
guacenc_buffer* buffer) {
|
|
|
|
/* If there is no decoder, simply return success */
|
|
guacenc_decoder* decoder = stream->decoder;
|
|
if (decoder == NULL)
|
|
return 0;
|
|
|
|
/* Decode received data to a Cairo surface */
|
|
cairo_surface_t* surface = stream->decoder(stream->buffer, stream->length);
|
|
if (surface == NULL)
|
|
return 1;
|
|
|
|
/* Get surface dimensions */
|
|
int width = cairo_image_surface_get_width(surface);
|
|
int height = cairo_image_surface_get_height(surface);
|
|
|
|
/* Expand the buffer as necessary to fit the draw operation */
|
|
if (buffer->autosize)
|
|
guacenc_buffer_fit(buffer, stream->x + width, stream->y + height);
|
|
|
|
/* Draw surface to buffer */
|
|
if (buffer->cairo != NULL) {
|
|
cairo_set_source_surface(buffer->cairo, surface, stream->x, stream->y);
|
|
cairo_rectangle(buffer->cairo, stream->x, stream->y, width, height);
|
|
cairo_fill(buffer->cairo);
|
|
}
|
|
|
|
cairo_surface_destroy(surface);
|
|
return 0;
|
|
|
|
}
|
|
|
|
int guacenc_image_stream_free(guacenc_image_stream* stream) {
|
|
|
|
/* Ignore NULL streams */
|
|
if (stream == NULL)
|
|
return 0;
|
|
|
|
/* Free image buffer */
|
|
free(stream->buffer);
|
|
|
|
/* Free actual stream */
|
|
free(stream);
|
|
return 0;
|
|
|
|
}
|
|
|