1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
|
#include <stdio.h>
#include <stdlib.h>
#include "config.h"
#include "mp_msg.h"
#include "vd_internal.h"
static vd_info_t info = {
"Quicktime Animation (RLE) decoder",
"qtrle",
"A'rpi",
"Mike Melanson",
"native codec"
};
LIBVD_EXTERN(qtrle)
typedef struct {
int depth;
void *palette;
} vd_qtrle_ctx;
// to set/get/query special features/parameters
static int control(sh_video_t *sh,int cmd,void* arg,...){
vd_qtrle_ctx *ctx = sh->context;
switch(cmd)
{
case VDCTRL_QUERY_FORMAT:
{
int req_format = *((int*)arg);
/* qtrle24 supports 32bit output too */
if ((req_format == (IMGFMT_BGR|ctx->depth)) ||
((IMGFMT_BGR_DEPTH(req_format) == 32) && (ctx->depth == 24)) ||
((IMGFMT_BGR_DEPTH(req_format) == 24) && (ctx->depth == 32)))
return(CONTROL_TRUE);
else
return(CONTROL_FALSE);
}
}
return CONTROL_UNKNOWN;
}
// init driver
static int init(sh_video_t *sh){
vd_qtrle_ctx *ctx;
ctx = sh->context = malloc(sizeof(vd_qtrle_ctx));
if (!ctx)
return(0);
memset(ctx, 0, sizeof(vd_qtrle_ctx));
if (!sh->bih)
return(0);
ctx->depth = sh->bih->biBitCount;
switch(ctx->depth)
{
case 2:
case 4:
case 8:
if (sh->bih->biSize > 40)
{
ctx->palette = malloc(sh->bih->biSize-40);
memcpy(ctx->palette, sh->bih+40, sh->bih->biSize-40);
}
break;
case 16:
ctx->depth--; /* this is the trick ;) */
break;
case 32:
mp_msg(MSGT_DECVIDEO,MSGL_INFO,"[qtrle] 32 bpp file, alpha channel will be ignored.\n");
case 24:
break;
default:
mp_msg(MSGT_DECVIDEO,MSGL_ERR,
"*** FYI: This Quicktime file is using %d-bit RLE Animation\n" \
"encoding, which is not yet supported by MPlayer. But if you upload\n" \
"this Quicktime file to the MPlayer FTP, the team could look at it.\n",
ctx->depth);
return(0);
}
return mpcodecs_config_vo(sh,sh->disp_w,sh->disp_h,IMGFMT_BGR|ctx->depth);
}
// uninit driver
static void uninit(sh_video_t *sh){
vd_qtrle_ctx *ctx = sh->context;
if (ctx->palette)
free(ctx->palette);
free(ctx);
}
//mp_image_t* mpcodecs_get_image(sh_video_t *sh, int mp_imgtype, int mp_imgflag, int w, int h);
void qt_decode_rle(
unsigned char *encoded,
int encoded_size,
unsigned char *decoded,
int width,
int height,
int encoded_bpp,
int bytes_per_pixel);
// decode a frame
static mp_image_t* decode(sh_video_t *sh,void* data,int len,int flags){
vd_qtrle_ctx *ctx = sh->context;
mp_image_t* mpi;
if(len<=0) return NULL; // skipped frame
mpi=mpcodecs_get_image(sh, MP_IMGTYPE_STATIC, MP_IMGFLAG_PRESERVE,
sh->disp_w, sh->disp_h);
if(!mpi) return NULL;
qt_decode_rle(
data,len, mpi->planes[0],
sh->disp_w, sh->disp_h,
sh->bih->biBitCount,
mpi->bpp/8);
if (ctx->palette)
mpi->planes[1] = ctx->palette;
return mpi;
}
|