diff options
Diffstat (limited to 'debian/transcode/transcode-1.1.7/encode/encode_lzo.c')
| -rw-r--r-- | debian/transcode/transcode-1.1.7/encode/encode_lzo.c | 293 |
1 files changed, 293 insertions, 0 deletions
diff --git a/debian/transcode/transcode-1.1.7/encode/encode_lzo.c b/debian/transcode/transcode-1.1.7/encode/encode_lzo.c new file mode 100644 index 00000000..b1200fbc --- /dev/null +++ b/debian/transcode/transcode-1.1.7/encode/encode_lzo.c @@ -0,0 +1,293 @@ +/* + * encode_lzo.c -- encode video frames individually using LZO. + * (C) 2005-2010 Francesco Romani <fromani at gmail dot com> + * + * This file is part of transcode, a video stream processing tool. + * + * transcode is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * transcode is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see <http://www.gnu.org/licenses/>. + */ + + +#include "transcode.h" +#include "aclib/imgconvert.h" +#include "libtc/optstr.h" +#include "libtc/tc_lzo.h" +#include "libtc/tcmodule-plugin.h" + +#include <stdio.h> +#include <stdlib.h> + +#define MOD_NAME "encode_lzo.so" +#define MOD_VERSION "v0.0.2 (2007-10-27)" +#define MOD_CAP "LZO lossless video encoder" + +#define MOD_FEATURES \ + TC_MODULE_FEATURE_ENCODE|TC_MODULE_FEATURE_VIDEO + +#define MOD_FLAGS \ + TC_MODULE_FLAG_RECONFIGURABLE + + +/* tc_lzo_ prefix was used to avoid any possible name clash with liblzo? */ + +static const char tc_lzo_help[] = "" + "Overview:\n" + " this module encodes raw RGB/YUV video frames in LZO, using liblzo V2.\n" + "Options:\n" + " help produce module overview and options explanations\n"; + +typedef struct { + lzo_byte work_mem[LZO1X_1_MEM_COMPRESS]; + /* needed by encoder to work properly */ + + int codec; + int flush_flag; +} LZOPrivateData; + +static int tc_lzo_configure(TCModuleInstance *self, + const char *options, vob_t *vob) +{ + LZOPrivateData *pd = NULL; + int ret; + + TC_MODULE_SELF_CHECK(self, "configure"); + + pd = self->userdata; + pd->codec = vob->im_v_codec; + pd->flush_flag = vob->encoder_flush; + + ret = lzo_init(); + if (ret != LZO_E_OK) { + tc_log_error(MOD_NAME, "configure: failed to initialize" + " LZO encoder"); + return TC_ERROR; + } + + return TC_OK; +} + +static int tc_lzo_stop(TCModuleInstance *self) +{ + LZOPrivateData *pd = NULL; + + TC_MODULE_SELF_CHECK(self, "stop"); + + pd = self->userdata; + + return TC_OK; +} + +static int tc_lzo_init(TCModuleInstance *self, uint32_t features) +{ + LZOPrivateData *pd = NULL; + + TC_MODULE_SELF_CHECK(self, "init"); + TC_MODULE_INIT_CHECK(self, MOD_FEATURES, features); + + pd = tc_malloc(sizeof(LZOPrivateData)); + if (!pd) { + tc_log_error(MOD_NAME, "init: can't allocate private data"); + return TC_ERROR; + } + /* sane defaults */ + pd->codec = CODEC_YUV; + + self->userdata = pd; + if (verbose) { + tc_log_info(MOD_NAME, "%s %s", MOD_VERSION, MOD_CAP); + } + return TC_OK; +} + +static int tc_lzo_fini(TCModuleInstance *self) +{ + TC_MODULE_SELF_CHECK(self, "fini"); + + tc_lzo_stop(self); + + tc_free(self->userdata); + self->userdata = NULL; + return TC_OK; +} + +static int tc_lzo_inspect(TCModuleInstance *self, + const char *param, const char **value) +{ + LZOPrivateData *pd = NULL; + + TC_MODULE_SELF_CHECK(self, "inspect"); + + pd = self->userdata; + + if (optstr_lookup(param, "help")) { + *value = tc_lzo_help; + } + + return TC_OK; +} + +/* ------------------------------------------------------------ + * + * encode and export + * + * ------------------------------------------------------------*/ + +/* assert(len(data) >= TC_LZ_HDR_SIZE) */ +static void tc_lzo_put_header(tc_lzo_header_t *hdr, void *data) +{ + /* always CPU byte order */ + uint32_t *ptr = data; + + *(ptr) = hdr->magic; + *(ptr + 1) = hdr->size; + *(ptr + 2) = hdr->flags; + *(ptr + 3) = (uint32_t)(hdr->method << 24 | hdr->level << 16 | hdr->pad); +} + +/* maybe translation should go away */ +static int tc_lzo_format_translate(int tc_codec) +{ + int ret; + switch (tc_codec) { + case CODEC_YUV: + ret = TC_LZO_FORMAT_YUV420P; + break; + case CODEC_YUY2: + ret = TC_LZO_FORMAT_YUY2; + break; + case CODEC_RGB: + ret = TC_LZO_FORMAT_RGB24; + break; + default: + /* shouldn't happen */ + ret = 0; + break; + } + return ret; +} + +static int tc_lzo_flush(TCModuleInstance *self, + vframe_list_t *outframe) +{ + outframe->video_len = 0; + return TC_OK; +} + + +static int tc_lzo_encode_video(TCModuleInstance *self, + vframe_list_t *inframe, vframe_list_t *outframe) +{ + LZOPrivateData *pd = NULL; + lzo_uint out_len = 0; + tc_lzo_header_t hdr; + int ret; + + TC_MODULE_SELF_CHECK(self, "encode_video"); + + pd = self->userdata; + + if (inframe == NULL && pd->flush_flag) { + return tc_lzo_flush(self, outframe); // FIXME + } + + /* invariants */ + hdr.magic = TC_CODEC_LZO2; + hdr.method = 1; + hdr.level = 1; + hdr.pad = 0; + hdr.flags = 0; /* sane default */ + + ret = lzo1x_1_compress(inframe->video_buf, inframe->video_size, + outframe->video_buf + TC_LZO_HDR_SIZE, + &out_len, pd->work_mem); + if (ret != LZO_E_OK) { + /* this should NEVER happen */ + tc_log_warn(MOD_NAME, "encode_video: LZO compression failed" + " (errcode=%i)", ret); + return TC_ERROR; + } + + /* check for an incompressible block */ + if (out_len >= inframe->video_size) { + hdr.flags |= TC_LZO_NOT_COMPRESSIBLE; + out_len = inframe->video_size; + } + hdr.size = out_len; + + hdr.flags |= tc_lzo_format_translate(pd->codec); + /* always put header */ + tc_lzo_put_header(&hdr, outframe->video_buf); + + if (hdr.flags & TC_LZO_NOT_COMPRESSIBLE) { + /* inframe data not compressible: outframe will hold a copy */ + if (verbose >= TC_DEBUG) { + tc_log_info(MOD_NAME, "encode_video: block contains" + " incompressible data"); + } + ac_memcpy(outframe->video_buf + TC_LZO_HDR_SIZE, + inframe->video_buf, out_len); + } else { + /* outframe data already in place */ + if (verbose >= TC_DEBUG) { + tc_log_info(MOD_NAME, "encode_video: compressed %lu bytes" + " into %lu bytes", + (unsigned long)inframe->video_size, + (unsigned long)out_len); + } + } + + /* only keyframes */ + outframe->video_len = out_len + TC_LZO_HDR_SIZE; + outframe->attributes |= TC_FRAME_IS_KEYFRAME; + + return TC_OK; +} + +/*************************************************************************/ + +static const TCCodecID tc_lzo_codecs_in[] = { + TC_CODEC_YUY2, TC_CODEC_RGB, TC_CODEC_YUV420P, TC_CODEC_ERROR +}; +static const TCCodecID tc_lzo_codecs_out[] = { + TC_CODEC_LZO2, TC_CODEC_ERROR +}; +TC_MODULE_CODEC_FORMATS(tc_lzo); + +TC_MODULE_INFO(tc_lzo); + +static const TCModuleClass tc_lzo_class = { + TC_MODULE_CLASS_HEAD(tc_lzo), + + .init = tc_lzo_init, + .fini = tc_lzo_fini, + .configure = tc_lzo_configure, + .stop = tc_lzo_stop, + .inspect = tc_lzo_inspect, + + .encode_video = tc_lzo_encode_video, +}; + +TC_MODULE_ENTRY_POINT(tc_lzo) + +/*************************************************************************/ + +/* + * Local variables: + * c-file-style: "stroustrup" + * c-file-offsets: ((case-label . *) (statement-case-intro . *)) + * indent-tabs-mode: nil + * End: + * + * vim: expandtab shiftwidth=4: + */ |
