/* * RemotelyAnywhere Screen Capture decoder * * Copyright (c) 2018 Paul B Mahol * * This file is part of FFmpeg. * * FFmpeg is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * FFmpeg 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with FFmpeg; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ #include <stdio.h> #include <stdlib.h> #include <string.h> #include "libavutil/avassert.h" #include "libavutil/imgutils.h" #include "libavutil/opt.h" #include "avcodec.h" #include "bytestream.h" #include "internal.h" #include <zlib.h> #define KBND MKTAG('K', 'B', 'N', 'D') #define FINT MKTAG('F', 'I', 'N', 'T') #define INIT MKTAG('I', 'N', 'I', 'T') #define BNDL MKTAG('B', 'N', 'D', 'L') #define KFRM MKTAG('K', 'F', 'R', 'M') #define DLTA MKTAG('D', 'L', 'T', 'A') #define MOUS MKTAG('M', 'O', 'U', 'S') #define MPOS MKTAG('M', 'P', 'O', 'S') #define MOVE MKTAG('M', 'O', 'V', 'E') #define EMPT MKTAG('E', 'M', 'P', 'T') typedef struct RASCContext { AVClass *class; int skip_cursor; GetByteContext gb; uint8_t *delta; int delta_size; uint8_t *cursor; int cursor_size; unsigned cursor_w; unsigned cursor_h; unsigned cursor_x; unsigned cursor_y; int stride; int bpp; z_stream zstream; AVFrame *frame; AVFrame *frame1; AVFrame *frame2; } RASCContext; static void clear_plane(AVCodecContext *avctx, AVFrame *frame) { RASCContext *s = avctx->priv_data; uint8_t *dst = frame->data[0]; for (int y = 0; y < avctx->height; y++) { memset(dst, 0, avctx->width * s->bpp); dst += frame->linesize[0]; } } static void copy_plane(AVCodecContext *avctx, AVFrame *src, AVFrame *dst) { RASCContext *s = avctx->priv_data; uint8_t *srcp = src->data[0]; uint8_t *dstp = dst->data[0]; for (int y = 0; y < avctx->height; y++) { memcpy(dstp, srcp, s->stride); srcp += src->linesize[0]; dstp += dst->linesize[0]; } } static int init_frames(AVCodecContext *avctx) { RASCContext *s = avctx->priv_data; int ret; av_frame_unref(s->frame1); av_frame_unref(s->frame2); if ((ret = ff_get_buffer(avctx, s->frame1, 0)) < 0) return ret; if ((ret = ff_get_buffer(avctx, s->frame2, 0)) < 0) return ret; clear_plane(avctx, s->frame2); clear_plane(avctx, s->frame1); return 0; } static int decode_fint(AVCodecContext *avctx, AVPacket *avpkt, unsigned size) { RASCContext *s = avctx->priv_data; GetByteContext *gb = &s->gb; unsigned w, h, fmt; int ret; if (bytestream2_peek_le32(gb) != 0x65) { if (!s->frame2->data[0] || !s->frame1->data[0]) return AVERROR_INVALIDDATA; clear_plane(avctx, s->frame2); clear_plane(avctx, s->frame1); return 0; } bytestream2_skip(gb, 8); w = bytestream2_get_le32(gb); h = bytestream2_get_le32(gb); bytestream2_skip(gb, 30); fmt = bytestream2_get_le16(gb); bytestream2_skip(gb, 24); switch (fmt) { case 8: s->stride = FFALIGN(w, 4); s->bpp = 1; fmt = AV_PIX_FMT_PAL8; break; case 16: s->stride = w * 2; s->bpp = 2; fmt = AV_PIX_FMT_RGB555LE; break; case 32: s->stride = w * 4; s->bpp = 4; fmt = AV_PIX_FMT_BGR0; break; default: return AVERROR_INVALIDDATA; } ret = ff_set_dimensions(avctx, w, h); if (ret < 0) return ret; avctx->width = w; avctx->height = h; avctx->pix_fmt = fmt; ret = init_frames(avctx); if (ret < 0) return ret; if (avctx->pix_fmt == AV_PIX_FMT_PAL8) { uint32_t *pal = (uint32_t *)s->frame2->data[1]; for (int i = 0; i < 256; i++) pal[i] = bytestream2_get_le32(gb) | 0xFF000000u; } return 0; } static int decode_zlib(AVCodecContext *avctx, AVPacket *avpkt, unsigned size, unsigned uncompressed_size) { RASCContext *s = avctx->priv_data; GetByteContext *gb = &s->gb; int zret; zret = inflateReset(&s->zstream); if (zret != Z_OK) { av_log(avctx, AV_LOG_ERROR, "Inflate reset error: %d\n", zret); return AVERROR_EXTERNAL; } av_fast_padded_malloc(&s->delta, &s->delta_size, uncompressed_size); if (!s->delta) return AVERROR(ENOMEM); s->zstream.next_in = avpkt->data + bytestream2_tell(gb); s->zstream.avail_in = FFMIN(size, bytestream2_get_bytes_left(gb)); s->zstream.next_out = s->delta; s->zstream.avail_out = s->delta_size; zret = inflate(&s->zstream, Z_FINISH); if (zret != Z_STREAM_END) { av_log(avctx, AV_LOG_ERROR, "Inflate failed with return code: %d.\n", zret); return AVERROR_INVALIDDATA; } return 0; } static int decode_move(AVCodecContext *avctx, AVPacket *avpkt, unsigned size) { RASCContext *s = avctx->priv_data; GetByteContext *gb = &s->gb; GetByteContext mc; unsigned pos, compression, nb_moves; unsigned uncompressed_size; int ret; pos = bytestream2_tell(gb); bytestream2_skip(gb, 8); nb_moves = bytestream2_get_le32(gb); bytestream2_skip(gb, 8); compression = bytestream2_get_le32(gb); if (nb_moves > INT32_MAX / 16 || nb_moves > avctx->width * avctx->height) return AVERROR_INVALIDDATA; uncompressed_size = 16 * nb_moves; if (compression == 1) { ret = decode_zlib(avctx, avpkt, size - (bytestream2_tell(gb) - pos), uncompressed_size); if (ret < 0) return ret; bytestream2_init(&mc, s->delta, uncompressed_size); } else if (compression == 0) { bytestream2_init(&mc, avpkt->data + bytestream2_tell(gb), bytestream2_get_bytes_left(gb)); } else if (compression == 2) { avpriv_request_sample(avctx, "compression %d", compression); return AVERROR_PATCHWELCOME; } else { return AVERROR_INVALIDDATA; } if (bytestream2_get_bytes_left(&mc) < uncompressed_size) return AVERROR_INVALIDDATA; for (int i = 0; i < nb_moves; i++) { int type, start_x, start_y, end_x, end_y, mov_x, mov_y; uint8_t *e2, *b1, *b2; int w, h; type = bytestream2_get_le16(&mc); start_x = bytestream2_get_le16(&mc); start_y = bytestream2_get_le16(&mc); end_x = bytestream2_get_le16(&mc); end_y = bytestream2_get_le16(&mc); mov_x = bytestream2_get_le16(&mc); mov_y = bytestream2_get_le16(&mc); bytestream2_skip(&mc, 2); if (start_x >= avctx->width || start_y >= avctx->height || end_x >= avctx->width || end_y >= avctx->height || mov_x >= avctx->width || mov_y >= avctx->height) { continue; } if (start_x >= end_x || start_y >= end_y) continue; w = end_x - start_x; h = end_y - start_y; if (mov_x + w > avctx->width || mov_y + h > avctx->height) continue; if (!s->frame2->data[0] || !s->frame1->data[0]) return AVERROR_INVALIDDATA; b1 = s->frame1->data[0] + s->frame1->linesize[0] * (start_y + h - 1) + start_x * s->bpp; b2 = s->frame2->data[0] + s->frame2->linesize[0] * (start_y + h - 1) + start_x * s->bpp; e2 = s->frame2->data[0] + s->frame2->linesize[0] * (mov_y + h - 1) + mov_x * s->bpp; if (type == 2) { for (int j = 0; j < h; j++) { memcpy(b1, b2, w * s->bpp); b1 -= s->frame1->linesize[0]; b2 -= s->frame2->linesize[0]; } } else if (type == 1) { for (int j = 0; j < h; j++) { memset(b2, 0, w * s->bpp); b2 -= s->frame2->linesize[0]; } } else if (type == 0) { uint8_t *buffer; av_fast_padded_malloc(&s->delta, &s->delta_size, w * h * s->bpp); buffer = s->delta; if (!buffer) return AVERROR(ENOMEM); for (int j = 0; j < h; j++) { memcpy(buffer + j * w * s->bpp, e2, w * s->bpp); e2 -= s->frame2->linesize[0]; } for (int j = 0; j < h; j++) { memcpy(b2, buffer + j * w * s->bpp, w * s->bpp); b2 -= s->frame2->linesize[0]; } } else { return AVERROR_INVALIDDATA; } } bytestream2_skip(gb, size - (bytestream2_tell(gb) - pos)); return 0; } #define NEXT_LINE \ if (cx >= w * s->bpp) { \ cx = 0; \ cy--; \ b1 -= s->frame1->linesize[0]; \ b2 -= s->frame2->linesize[0]; \ } \ len--; static int decode_dlta(AVCodecContext *avctx, AVPacket *avpkt, unsigned size) { RASCContext *s = avctx->priv_data; GetByteContext *gb = &s->gb; GetByteContext dc; unsigned uncompressed_size, pos; unsigned x, y, w, h; int ret, cx, cy, compression; uint8_t *b1, *b2; pos = bytestream2_tell(gb); bytestream2_skip(gb, 12); uncompressed_size = bytestream2_get_le32(gb); x = bytestream2_get_le32(gb); y = bytestream2_get_le32(gb); w = bytestream2_get_le32(gb); h = bytestream2_get_le32(gb); if (x >= avctx->width || y >= avctx->height || w > avctx->width || h > avctx->height) return AVERROR_INVALIDDATA; if (x + w > avctx->width || y + h > avctx->height) return AVERROR_INVALIDDATA; bytestream2_skip(gb, 4); compression = bytestream2_get_le32(gb); if (compression == 1) { if (w * h * s->bpp * 3 < uncompressed_size) return AVERROR_INVALIDDATA; ret = decode_zlib(avctx, avpkt, size, uncompressed_size); if (ret < 0) return ret; bytestream2_init(&dc, s->delta, uncompressed_size); } else if (compression == 0) { if (bytestream2_get_bytes_left(gb) < uncompressed_size) return AVERROR_INVALIDDATA; bytestream2_init(&dc, avpkt->data + bytestream2_tell(gb), uncompressed_size); } else if (compression == 2) { avpriv_request_sample(avctx, "compression %d", compression); return AVERROR_PATCHWELCOME; } else { return AVERROR_INVALIDDATA; } if (!s->frame2->data[0] || !s->frame1->data[0]) return AVERROR_INVALIDDATA; b1 = s->frame1->data[0] + s->frame1->linesize[0] * (y + h - 1) + x * s->bpp; b2 = s->frame2->data[0] + s->frame2->linesize[0] * (y + h - 1) + x * s->bpp; cx = 0, cy = h; while (bytestream2_get_bytes_left(&dc) > 0) { int type = bytestream2_get_byte(&dc); int len = bytestream2_get_byte(&dc); unsigned fill; switch (type) { case 1: while (len > 0 && cy > 0) { cx++; NEXT_LINE } break; case 2: while (len > 0 && cy > 0) { int v0 = b1[cx]; int v1 = b2[cx]; b2[cx] = v0; b1[cx] = v1; cx++; NEXT_LINE } break; case 3: while (len > 0 && cy > 0) { fill = bytestream2_get_byte(&dc); b1[cx] = b2[cx]; b2[cx] = fill; cx++; NEXT_LINE } break; case 4: fill = bytestream2_get_byte(&dc); while (len > 0 && cy > 0) { AV_WL32(b1 + cx, AV_RL32(b2 + cx)); AV_WL32(b2 + cx, fill); cx++; NEXT_LINE } break; case 7: fill = bytestream2_get_le32(&dc); while (len > 0 && cy > 0) { AV_WL32(b1 + cx, AV_RL32(b2 + cx)); AV_WL32(b2 + cx, fill); cx += 4; NEXT_LINE } break; case 10: while (len > 0 && cy > 0) { cx += 4; NEXT_LINE } break; case 12: while (len > 0 && cy > 0) { unsigned v0, v1; v0 = AV_RL32(b2 + cx); v1 = AV_RL32(b1 + cx); AV_WL32(b2 + cx, v1); AV_WL32(b1 + cx, v0); cx += 4; NEXT_LINE } break; case 13: while (len > 0 && cy > 0) { fill = bytestream2_get_le32(&dc); AV_WL32(b1 + cx, AV_RL32(b2 + cx)); AV_WL32(b2 + cx, fill); cx += 4; NEXT_LINE } break; default: avpriv_request_sample(avctx, "runlen %d", type); return AVERROR_INVALIDDATA; } } bytestream2_skip(gb, size - (bytestream2_tell(gb) - pos)); return 0; } static int decode_kfrm(AVCodecContext *avctx, AVPacket *avpkt, unsigned size) { RASCContext *s = avctx->priv_data; GetByteContext *gb = &s->gb; uint8_t *dst; unsigned pos; int zret, ret; pos = bytestream2_tell(gb); if (bytestream2_peek_le32(gb) == 0x65) { ret = decode_fint(avctx, avpkt, size); if (ret < 0) return ret; } if (!s->frame2->data[0]) return AVERROR_INVALIDDATA; zret = inflateReset(&s->zstream); if (zret != Z_OK) { av_log(avctx, AV_LOG_ERROR, "Inflate reset error: %d\n", zret); return AVERROR_EXTERNAL; } s->zstream.next_in = avpkt->data + bytestream2_tell(gb); s->zstream.avail_in = bytestream2_get_bytes_left(gb); dst = s->frame2->data[0] + (avctx->height - 1) * s->frame2->linesize[0]; for (int i = 0; i < avctx->height; i++) { s->zstream.next_out = dst; s->zstream.avail_out = s->stride; zret = inflate(&s->zstream, Z_SYNC_FLUSH); if (zret != Z_OK && zret != Z_STREAM_END) { av_log(avctx, AV_LOG_ERROR, "Inflate failed with return code: %d.\n", zret); return AVERROR_INVALIDDATA; } dst -= s->frame2->linesize[0]; } dst = s->frame1->data[0] + (avctx->height - 1) * s->frame1->linesize[0]; for (int i = 0; i < avctx->height; i++) { s->zstream.next_out = dst; s->zstream.avail_out = s->stride; zret = inflate(&s->zstream, Z_SYNC_FLUSH); if (zret != Z_OK && zret != Z_STREAM_END) { av_log(avctx, AV_LOG_ERROR, "Inflate failed with return code: %d.\n", zret); return AVERROR_INVALIDDATA; } dst -= s->frame1->linesize[0]; } bytestream2_skip(gb, size - (bytestream2_tell(gb) - pos)); return 0; } static int decode_mous(AVCodecContext *avctx, AVPacket *avpkt, unsigned size) { RASCContext *s = avctx->priv_data; GetByteContext *gb = &s->gb; unsigned w, h, pos, uncompressed_size; int ret; pos = bytestream2_tell(gb); bytestream2_skip(gb, 8); w = bytestream2_get_le32(gb); h = bytestream2_get_le32(gb); bytestream2_skip(gb, 12); uncompressed_size = bytestream2_get_le32(gb); if (w > avctx->width || h > avctx->height) return AVERROR_INVALIDDATA; if (uncompressed_size != 3 * w * h) return AVERROR_INVALIDDATA; av_fast_padded_malloc(&s->cursor, &s->cursor_size, uncompressed_size); if (!s->cursor) return AVERROR(ENOMEM); ret = decode_zlib(avctx, avpkt, size - (bytestream2_tell(gb) - pos), uncompressed_size); if (ret < 0) return ret; memcpy(s->cursor, s->delta, uncompressed_size); bytestream2_skip(gb, size - (bytestream2_tell(gb) - pos)); s->cursor_w = w; s->cursor_h = h; return 0; } static int decode_mpos(AVCodecContext *avctx, AVPacket *avpkt, unsigned size) { RASCContext *s = avctx->priv_data; GetByteContext *gb = &s->gb; unsigned pos; pos = bytestream2_tell(gb); bytestream2_skip(gb, 8); s->cursor_x = bytestream2_get_le32(gb); s->cursor_y = bytestream2_get_le32(gb); bytestream2_skip(gb, size - (bytestream2_tell(gb) - pos)); return 0; } static void draw_cursor(AVCodecContext *avctx) { RASCContext *s = avctx->priv_data; uint8_t *dst, *pal; if (!s->cursor) return; if (s->cursor_x >= avctx->width || s->cursor_y >= avctx->height) return; if (s->cursor_x + s->cursor_w > avctx->width || s->cursor_y + s->cursor_h > avctx->height) return; if (avctx->pix_fmt == AV_PIX_FMT_PAL8) { pal = s->frame->data[1]; for (int i = 0; i < s->cursor_h; i++) { for (int j = 0; j < s->cursor_w; j++) { int cr = s->cursor[3 * s->cursor_w * (s->cursor_h - i - 1) + 3 * j + 0]; int cg = s->cursor[3 * s->cursor_w * (s->cursor_h - i - 1) + 3 * j + 1]; int cb = s->cursor[3 * s->cursor_w * (s->cursor_h - i - 1) + 3 * j + 2]; int best = INT_MAX; int index = 0; int dist; if (cr == s->cursor[0] && cg == s->cursor[1] && cb == s->cursor[2]) continue; dst = s->frame->data[0] + s->frame->linesize[0] * (s->cursor_y + i) + (s->cursor_x + j); for (int k = 0; k < 256; k++) { int pr = pal[k * 4 + 0]; int pg = pal[k * 4 + 1]; int pb = pal[k * 4 + 2]; dist = FFABS(cr - pr) + FFABS(cg - pg) + FFABS(cb - pb); if (dist < best) { best = dist; index = k; } } dst[0] = index; } } } else if (avctx->pix_fmt == AV_PIX_FMT_RGB555LE) { for (int i = 0; i < s->cursor_h; i++) { for (int j = 0; j < s->cursor_w; j++) { int cr = s->cursor[3 * s->cursor_w * (s->cursor_h - i - 1) + 3 * j + 0]; int cg = s->cursor[3 * s->cursor_w * (s->cursor_h - i - 1) + 3 * j + 1]; int cb = s->cursor[3 * s->cursor_w * (s->cursor_h - i - 1) + 3 * j + 2]; if (cr == s->cursor[0] && cg == s->cursor[1] && cb == s->cursor[2]) continue; cr >>= 3; cg >>=3; cb >>= 3; dst = s->frame->data[0] + s->frame->linesize[0] * (s->cursor_y + i) + 2 * (s->cursor_x + j); AV_WL16(dst, cr | cg << 5 | cb << 10); } } } else if (avctx->pix_fmt == AV_PIX_FMT_BGR0) { for (int i = 0; i < s->cursor_h; i++) { for (int j = 0; j < s->cursor_w; j++) { int cr = s->cursor[3 * s->cursor_w * (s->cursor_h - i - 1) + 3 * j + 0]; int cg = s->cursor[3 * s->cursor_w * (s->cursor_h - i - 1) + 3 * j + 1]; int cb = s->cursor[3 * s->cursor_w * (s->cursor_h - i - 1) + 3 * j + 2]; if (cr == s->cursor[0] && cg == s->cursor[1] && cb == s->cursor[2]) continue; dst = s->frame->data[0] + s->frame->linesize[0] * (s->cursor_y + i) + 4 * (s->cursor_x + j); dst[0] = cb; dst[1] = cg; dst[2] = cr; } } } } static int decode_frame(AVCodecContext *avctx, void *data, int *got_frame, AVPacket *avpkt) { RASCContext *s = avctx->priv_data; GetByteContext *gb = &s->gb; int ret, intra = 0; AVFrame *frame = data; bytestream2_init(gb, avpkt->data, avpkt->size); if (bytestream2_peek_le32(gb) == EMPT) return avpkt->size; s->frame = frame; while (bytestream2_get_bytes_left(gb) > 0) { unsigned type, size = 0; if (bytestream2_get_bytes_left(gb) < 8) return AVERROR_INVALIDDATA; type = bytestream2_get_le32(gb); if (type == KBND || type == BNDL) { intra = type == KBND; type = bytestream2_get_le32(gb); } size = bytestream2_get_le32(gb); if (bytestream2_get_bytes_left(gb) < size) return AVERROR_INVALIDDATA; switch (type) { case FINT: case INIT: ret = decode_fint(avctx, avpkt, size); break; case KFRM: ret = decode_kfrm(avctx, avpkt, size); break; case DLTA: ret = decode_dlta(avctx, avpkt, size); break; case MOVE: ret = decode_move(avctx, avpkt, size); break; case MOUS: ret = decode_mous(avctx, avpkt, size); break; case MPOS: ret = decode_mpos(avctx, avpkt, size); break; default: bytestream2_skip(gb, size); } if (ret < 0) return ret; } if (!s->frame2->data[0] || !s->frame1->data[0]) return AVERROR_INVALIDDATA; if ((ret = ff_get_buffer(avctx, s->frame, 0)) < 0) return ret; copy_plane(avctx, s->frame2, s->frame); if (avctx->pix_fmt == AV_PIX_FMT_PAL8) memcpy(s->frame->data[1], s->frame2->data[1], 1024); if (!s->skip_cursor) draw_cursor(avctx); s->frame->key_frame = intra; s->frame->pict_type = intra ? AV_PICTURE_TYPE_I : AV_PICTURE_TYPE_P; *got_frame = 1; return avpkt->size; } static av_cold int decode_init(AVCodecContext *avctx) { RASCContext *s = avctx->priv_data; int zret; s->zstream.zalloc = Z_NULL; s->zstream.zfree = Z_NULL; s->zstream.opaque = Z_NULL; zret = inflateInit(&s->zstream); if (zret != Z_OK) { av_log(avctx, AV_LOG_ERROR, "Inflate init error: %d\n", zret); return AVERROR_EXTERNAL; } s->frame1 = av_frame_alloc(); s->frame2 = av_frame_alloc(); if (!s->frame1 || !s->frame2) return AVERROR(ENOMEM); return 0; } static av_cold int decode_close(AVCodecContext *avctx) { RASCContext *s = avctx->priv_data; av_freep(&s->cursor); s->cursor_size = 0; av_freep(&s->delta); s->delta_size = 0; av_frame_free(&s->frame1); av_frame_free(&s->frame2); inflateEnd(&s->zstream); return 0; } static void decode_flush(AVCodecContext *avctx) { RASCContext *s = avctx->priv_data; clear_plane(avctx, s->frame1); clear_plane(avctx, s->frame2); } static const AVOption options[] = { { "skip_cursor", "skip the cursor", offsetof(RASCContext, skip_cursor), AV_OPT_TYPE_BOOL, {.i64 = 0 }, 0, 1, AV_OPT_FLAG_DECODING_PARAM | AV_OPT_FLAG_VIDEO_PARAM }, { NULL }, }; static const AVClass rasc_decoder_class = { .class_name = "rasc decoder", .item_name = av_default_item_name, .option = options, .version = LIBAVUTIL_VERSION_INT, }; AVCodec ff_rasc_decoder = { .name = "rasc", .long_name = NULL_IF_CONFIG_SMALL("RemotelyAnywhere Screen Capture"), .type = AVMEDIA_TYPE_VIDEO, .id = AV_CODEC_ID_RASC, .priv_data_size = sizeof(RASCContext), .init = decode_init, .close = decode_close, .decode = decode_frame, .flush = decode_flush, .capabilities = AV_CODEC_CAP_DR1, .caps_internal = FF_CODEC_CAP_INIT_THREADSAFE | FF_CODEC_CAP_INIT_CLEANUP, .priv_class = &rasc_decoder_class, };