2001-02-24 21:28:24 +01:00
|
|
|
/*
|
|
|
|
* OpenDivX AVI file writer
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2002-02-17 09:24:43 +01:00
|
|
|
#include <errno.h>
|
2001-02-24 21:28:24 +01:00
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
#include "video_out.h"
|
|
|
|
#include "video_out_internal.h"
|
|
|
|
|
|
|
|
LIBVO_EXTERN(odivx)
|
|
|
|
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <sys/mman.h>
|
|
|
|
|
|
|
|
#include "../encore/encore.h"
|
|
|
|
|
2001-04-11 21:41:38 +02:00
|
|
|
#include "fastmemcpy.h"
|
2001-04-11 14:47:45 +02:00
|
|
|
|
2001-02-24 21:28:24 +01:00
|
|
|
static vo_info_t vo_info =
|
|
|
|
{
|
|
|
|
"OpenDivX AVI File writer",
|
|
|
|
"odivx",
|
|
|
|
"Arpad Gereoffy <arpi@esp-team.scene.hu>",
|
|
|
|
""
|
|
|
|
};
|
|
|
|
|
|
|
|
static uint8_t *image=NULL;
|
|
|
|
static int image_width=0;
|
|
|
|
static int image_height=0;
|
|
|
|
static unsigned int image_format=0;
|
|
|
|
static char *buffer=NULL;
|
|
|
|
static int frameno=0;
|
|
|
|
|
|
|
|
extern char* encode_name;
|
|
|
|
extern char* encode_index_name;
|
|
|
|
|
|
|
|
//static uint32_t draw_slice(uint8_t *src[], uint32_t slice_num)
|
|
|
|
static uint32_t draw_slice(uint8_t *src[], int stride[], int w,int h,int x,int y)
|
|
|
|
{
|
|
|
|
uint8_t *s;
|
|
|
|
uint8_t *d;
|
|
|
|
int i;
|
|
|
|
int dstride=image_width;
|
|
|
|
|
|
|
|
// copy Y
|
|
|
|
d=image+dstride*y+x;
|
|
|
|
s=src[0];
|
|
|
|
for(i=0;i<h;i++){
|
|
|
|
memcpy(d,s,w);
|
|
|
|
s+=stride[0];
|
|
|
|
d+=dstride;
|
|
|
|
}
|
|
|
|
|
|
|
|
w/=2;h/=2;x/=2;y/=2; dstride/=2;
|
|
|
|
|
|
|
|
// copy U
|
|
|
|
d=image+image_width*image_height + dstride*y+x;
|
|
|
|
s=src[1];
|
|
|
|
for(i=0;i<h;i++){
|
|
|
|
memcpy(d,s,w);
|
|
|
|
s+=stride[1];
|
|
|
|
d+=dstride;
|
|
|
|
}
|
|
|
|
|
|
|
|
// copy V
|
|
|
|
d=image+image_width*image_height +image_width*image_height/4 + dstride*y+x;
|
|
|
|
s=src[2];
|
|
|
|
for(i=0;i<h;i++){
|
|
|
|
memcpy(d,s,w);
|
|
|
|
s+=stride[2];
|
|
|
|
d+=dstride;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint32_t
|
|
|
|
draw_frame(uint8_t *src[])
|
|
|
|
{
|
|
|
|
uint8_t *d=image;
|
|
|
|
|
|
|
|
switch(image_format){
|
|
|
|
case IMGFMT_YV12:
|
|
|
|
// copy Y
|
|
|
|
memcpy(d,src[0],image_width*image_height);
|
|
|
|
// copy U
|
|
|
|
d+=image_width*image_height;
|
|
|
|
memcpy(d,src[1],image_width*image_height/4);
|
|
|
|
// copy V
|
|
|
|
d+=image_width*image_height/4;
|
|
|
|
memcpy(d,src[2],image_width*image_height/4);
|
|
|
|
break;
|
2001-03-11 03:45:02 +01:00
|
|
|
case IMGFMT_YUY2: {
|
|
|
|
uint8_t *dY=image;
|
|
|
|
uint8_t *dU=image+image_width*image_height;
|
|
|
|
uint8_t *dV=dU+image_width*image_height/4;
|
|
|
|
uint8_t *s=src[0];
|
|
|
|
int y;
|
|
|
|
for(y=0;y<image_height;y+=2){
|
|
|
|
uint8_t *e=s+image_width*2;
|
|
|
|
while(s<e){
|
|
|
|
*dY++=s[0];
|
|
|
|
*dU++=s[1];
|
|
|
|
*dY++=s[2];
|
|
|
|
*dV++=s[3];
|
|
|
|
s+=4;
|
|
|
|
}
|
|
|
|
e=s+image_width*2;
|
|
|
|
while(s<e){
|
|
|
|
*dY++=s[0];
|
|
|
|
*dY++=s[2];
|
|
|
|
s+=4;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// case IMGFMT_BGR|24:
|
|
|
|
// memcpy(d,src[0],image_width*image_height*2);
|
2001-02-24 21:28:24 +01:00
|
|
|
break;
|
|
|
|
}
|
2001-03-11 03:45:02 +01:00
|
|
|
}
|
2001-02-24 21:28:24 +01:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef unsigned int DWORD;
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
DWORD ckid;
|
|
|
|
DWORD dwFlags;
|
|
|
|
DWORD dwChunkOffset; // Position of chunk
|
|
|
|
DWORD dwChunkLength; // Length of chunk
|
|
|
|
} AVIINDEXENTRY;
|
|
|
|
|
2001-08-13 13:33:34 +02:00
|
|
|
static void draw_osd(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2001-02-24 21:28:24 +01:00
|
|
|
static void
|
|
|
|
flip_page(void)
|
|
|
|
{
|
|
|
|
|
|
|
|
// we are rady to encode this frame
|
|
|
|
ENC_FRAME enc_frame;
|
|
|
|
ENC_RESULT enc_result;
|
|
|
|
|
|
|
|
if(++frameno<10) return;
|
|
|
|
|
2001-03-11 03:45:02 +01:00
|
|
|
enc_frame.image=image;
|
2001-02-24 21:28:24 +01:00
|
|
|
enc_frame.bitstream=buffer;
|
|
|
|
enc_frame.length=0;
|
|
|
|
encore(0x123,0,&enc_frame,&enc_result);
|
|
|
|
|
2001-04-24 12:21:12 +02:00
|
|
|
printf("coded length: %ld \n",enc_frame.length);
|
2001-02-24 21:28:24 +01:00
|
|
|
|
|
|
|
if(encode_name){
|
|
|
|
AVIINDEXENTRY i;
|
|
|
|
FILE *file;
|
|
|
|
i.ckid=('c'<<24)|('d'<<16)|('0'<<8)|'0'; // "00dc"
|
|
|
|
i.dwFlags=enc_result.isKeyFrame?0x10:0;
|
|
|
|
i.dwChunkLength=enc_frame.length;
|
|
|
|
// Write AVI chunk:
|
|
|
|
if((file=fopen(encode_name,"ab"))){
|
|
|
|
unsigned char zerobyte=0;
|
|
|
|
i.dwChunkOffset=ftell(file);
|
|
|
|
fwrite(&i.ckid,4,1,file);
|
|
|
|
fwrite(&enc_frame.length,4,1,file);
|
|
|
|
fwrite(buffer,enc_frame.length,1,file);
|
|
|
|
if(enc_frame.length&1) fwrite(&zerobyte,1,1,file); // padding
|
|
|
|
fclose(file);
|
|
|
|
}
|
|
|
|
// Write AVI index:
|
|
|
|
if(encode_index_name && (file=fopen(encode_index_name,"ab"))){
|
|
|
|
fwrite(&i,sizeof(i),1,file);
|
|
|
|
fclose(file);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint32_t
|
|
|
|
query_format(uint32_t format)
|
|
|
|
{
|
|
|
|
switch(format){
|
|
|
|
case IMGFMT_YV12:
|
2001-03-11 03:45:02 +01:00
|
|
|
case IMGFMT_YUY2:
|
|
|
|
// case IMGFMT_BGR|24:
|
2001-02-24 21:28:24 +01:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern int encode_bitrate;
|
|
|
|
|
|
|
|
static uint32_t
|
2002-01-31 10:57:13 +01:00
|
|
|
config(uint32_t width, uint32_t height, uint32_t d_width, uint32_t d_height, uint32_t fullscreen, char *title, uint32_t format,const vo_tune_info_t *info)
|
2001-02-24 21:28:24 +01:00
|
|
|
{
|
|
|
|
uint32_t frame_size;
|
|
|
|
ENC_PARAM enc_param;
|
|
|
|
|
|
|
|
// file = fopen("encoded.odvx","wb");
|
|
|
|
// if(!file) return -1;
|
|
|
|
|
|
|
|
switch(format){
|
|
|
|
case IMGFMT_YV12:
|
|
|
|
frame_size=width*height+width*height/2;
|
2001-03-11 03:45:02 +01:00
|
|
|
// enc_param.flip=2; // 0=RGB 1=flippedRGB 2=planarYUV format
|
2001-02-24 21:28:24 +01:00
|
|
|
break;
|
2001-03-11 03:45:02 +01:00
|
|
|
case IMGFMT_YUY2:
|
|
|
|
// case IMGFMT_BGR|24:
|
|
|
|
// enc_param.flip=0; // 0=RGB 1=flippedRGB 2=planarYUV format
|
|
|
|
// frame_size=width*height*2;
|
|
|
|
frame_size=width*height+width*height/2;
|
2001-02-24 21:28:24 +01:00
|
|
|
break;
|
|
|
|
default: return -1; // invalid format
|
|
|
|
}
|
|
|
|
|
|
|
|
enc_param.x_dim=width;
|
|
|
|
enc_param.y_dim=height;
|
|
|
|
|
|
|
|
image_width=width;
|
|
|
|
image_height=height;
|
|
|
|
image_format=format;
|
|
|
|
image=malloc(frame_size);
|
|
|
|
|
|
|
|
//clear the buffer
|
|
|
|
memset(image,0x80,frame_size);
|
|
|
|
|
|
|
|
// buffer for encoded video data:
|
|
|
|
buffer=malloc(0x100000);
|
|
|
|
if(!buffer) return -1;
|
|
|
|
|
|
|
|
// encoding parameters:
|
|
|
|
enc_param.framerate=25.0;
|
|
|
|
enc_param.bitrate=encode_bitrate?encode_bitrate:780000;
|
|
|
|
enc_param.rc_period=300;
|
|
|
|
enc_param.max_quantizer=15;
|
|
|
|
enc_param.min_quantizer=1;
|
|
|
|
enc_param.search_range=128;
|
|
|
|
|
|
|
|
// init codec:
|
|
|
|
encore(0x123,ENC_OPT_INIT,&enc_param,NULL);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const vo_info_t*
|
|
|
|
get_info(void)
|
|
|
|
{
|
|
|
|
return &vo_info;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
uninit(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2001-03-03 22:46:39 +01:00
|
|
|
static void check_events(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2002-01-26 17:01:26 +01:00
|
|
|
static uint32_t preinit(const char *arg)
|
|
|
|
{
|
2002-02-17 09:24:43 +01:00
|
|
|
if(arg)
|
|
|
|
{
|
|
|
|
printf("vo_odivx: Unknown subdevice: %s\n",arg);
|
|
|
|
return ENOSYS;
|
|
|
|
}
|
|
|
|
return 0;
|
2002-01-26 17:01:26 +01:00
|
|
|
}
|
2001-03-03 22:46:39 +01:00
|
|
|
|
2002-02-09 02:21:48 +01:00
|
|
|
static uint32_t control(uint32_t request, void *data, ...)
|
2002-01-26 17:01:26 +01:00
|
|
|
{
|
2002-02-09 01:47:26 +01:00
|
|
|
switch (request) {
|
|
|
|
case VOCTRL_QUERY_FORMAT:
|
|
|
|
return query_format(*((uint32_t*)data));
|
|
|
|
}
|
|
|
|
return VO_NOTIMPL;
|
2002-01-26 17:01:26 +01:00
|
|
|
}
|