summaryrefslogtreecommitdiffstats
path: root/debian/transcode/transcode-1.1.7/encode/encode_lzo.c
diff options
context:
space:
mode:
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.c293
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:
+ */