diff options
Diffstat (limited to 'libmpdemux')
-rw-r--r-- | libmpdemux/muxer.c | 167 | ||||
-rw-r--r-- | libmpdemux/muxer.h | 143 | ||||
-rw-r--r-- | libmpdemux/muxer_avi.c | 694 | ||||
-rw-r--r-- | libmpdemux/muxer_lavf.c | 408 | ||||
-rw-r--r-- | libmpdemux/muxer_mpeg.c | 2735 | ||||
-rw-r--r-- | libmpdemux/muxer_rawaudio.c | 89 | ||||
-rw-r--r-- | libmpdemux/muxer_rawvideo.c | 92 |
7 files changed, 0 insertions, 4328 deletions
diff --git a/libmpdemux/muxer.c b/libmpdemux/muxer.c deleted file mode 100644 index 32ac1bef8c..0000000000 --- a/libmpdemux/muxer.c +++ /dev/null @@ -1,167 +0,0 @@ -/* - * This file is part of MPlayer. - * - * MPlayer 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. - * - * MPlayer 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 MPlayer; 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 <inttypes.h> -#include <unistd.h> - -#include "config.h" -#include "aviheader.h" -#include "ms_hdr.h" - -#include "stream/stream.h" -#include "muxer.h" -#include "demuxer.h" -#include "mp_msg.h" -#include "stheader.h" - -muxer_t *muxer_new_muxer(int type,stream_t *stream){ - muxer_t* muxer=calloc(1,sizeof(muxer_t)); - if(!muxer) - return NULL; - muxer->stream = stream; - switch (type) { - case MUXER_TYPE_MPEG: - if(! muxer_init_muxer_mpeg(muxer)) - goto fail; - break; - case MUXER_TYPE_RAWVIDEO: - if(! muxer_init_muxer_rawvideo(muxer)) - goto fail; - break; - case MUXER_TYPE_RAWAUDIO: - if(! muxer_init_muxer_rawaudio(muxer)) - goto fail; - break; -#ifdef CONFIG_LIBAVFORMAT - case MUXER_TYPE_LAVF: - if(! muxer_init_muxer_lavf(muxer)) - goto fail; - break; -#endif - case MUXER_TYPE_AVI: - default: - if(! muxer_init_muxer_avi(muxer)) - goto fail; - } - return muxer; - -fail: - free(muxer); - return NULL; -} - -/* buffer frames until we either: - * (a) have at least one frame from each stream - * (b) run out of memory */ -void muxer_write_chunk(muxer_stream_t *s, size_t len, unsigned int flags, double dts, double pts) { - if(dts == MP_NOPTS_VALUE) dts= s->timer; - if(pts == MP_NOPTS_VALUE) pts= s->timer; // this is wrong - - if (s->muxer->muxbuf_skip_buffer) { - s->muxer->cont_write_chunk(s, len, flags, dts, pts); - } - else { - int num = s->muxer->muxbuf_num++; - muxbuf_t *buf, *tmp; - - tmp = realloc_struct(s->muxer->muxbuf, (num+1), sizeof(muxbuf_t)); - if(!tmp) { - mp_tmsg(MSGT_MUXER, MSGL_FATAL, "Muxer frame buffer cannot reallocate memory!\n"); - return; - } - s->muxer->muxbuf = tmp; - buf = s->muxer->muxbuf + num; - - /* buffer this frame */ - buf->stream = s; - buf->dts= dts; - buf->pts= pts; - buf->len = len; - buf->flags = flags; - buf->buffer = malloc(len); - if (!buf->buffer) { - mp_tmsg(MSGT_MUXER, MSGL_FATAL, "Muxer frame buffer cannot allocate memory!\n"); - return; - } - memcpy(buf->buffer, s->buffer, buf->len); - s->muxbuf_seen = 1; - - /* see if we need to keep buffering */ - s->muxer->muxbuf_skip_buffer = 1; - for (num = 0; s->muxer->streams[num]; ++num) - if (!s->muxer->streams[num]->muxbuf_seen) - s->muxer->muxbuf_skip_buffer = 0; - - /* see if we can flush buffer now */ - if (s->muxer->muxbuf_skip_buffer) { - mp_tmsg(MSGT_MUXER, MSGL_V, "Muxer frame buffer sending %d frame(s) to the muxer.\n", s->muxer->muxbuf_num); - - /* fix parameters for all streams */ - for (num = 0; s->muxer->streams[num]; ++num) { - muxer_stream_t *str = s->muxer->streams[num]; - if(str->muxer->fix_stream_parameters) - muxer_stream_fix_parameters(str->muxer, str); - } - - /* write header */ - if (s->muxer->cont_write_header) - muxer_write_header(s->muxer); - - /* send all buffered frames to muxer */ - for (num = 0; num < s->muxer->muxbuf_num; ++num) { - muxbuf_t tmp_buf; - buf = s->muxer->muxbuf + num; - s = buf->stream; - - /* 1. save timer and buffer (might have changed by now) */ - tmp_buf.dts = s->timer; - tmp_buf.buffer = s->buffer; - - /* 2. move stored timer and buffer into stream and mux it */ - s->timer = buf->dts; - s->buffer = buf->buffer; - s->muxer->cont_write_chunk(s, buf->len, buf->flags, buf->dts, buf->pts); - - /* 3. restore saved timer and buffer */ - s->timer = tmp_buf.dts; - s->buffer = tmp_buf.buffer; - } - - free(s->muxer->muxbuf); - s->muxer->muxbuf_num = 0; - } - } - - /* this code moved directly from muxer_avi.c */ - // alter counters: - if(s->h.dwSampleSize){ - // CBR - s->h.dwLength+=len/s->h.dwSampleSize; - if(len%s->h.dwSampleSize) mp_tmsg(MSGT_MUXER, MSGL_WARN, "Warning, len isn't divisible by samplesize!\n"); - } else { - // VBR - s->h.dwLength++; - } - s->timer=(double)s->h.dwLength*s->h.dwScale/s->h.dwRate; - s->size+=len; - - return; -} diff --git a/libmpdemux/muxer.h b/libmpdemux/muxer.h deleted file mode 100644 index 48b132b85f..0000000000 --- a/libmpdemux/muxer.h +++ /dev/null @@ -1,143 +0,0 @@ -/* - * This file is part of MPlayer. - * - * MPlayer 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. - * - * MPlayer 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 MPlayer; if not, write to the Free Software Foundation, Inc., - * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. - */ - -#ifndef MPLAYER_MUXER_H -#define MPLAYER_MUXER_H - -#include <stdint.h> -#include <sys/types.h> -#include "aviheader.h" -#include "ms_hdr.h" -#include "stream/stream.h" - -#define MUXER_MAX_STREAMS 16 - -#define MUXER_TYPE_VIDEO 0 -#define MUXER_TYPE_AUDIO 1 - -#define MUXER_TYPE_AVI 0 -#define MUXER_TYPE_MPEG 1 -#define MUXER_TYPE_RAWVIDEO 2 -#define MUXER_TYPE_LAVF 3 -#define MUXER_TYPE_RAWAUDIO 4 - - -typedef struct { - // muxer data: - int type; // audio or video - int id; // stream no - uint32_t ckid; // chunk id (00dc 01wb etc) - double timer; - off_t size; - float aspect; // aspect ratio of this stream (set by ve_*.c) - // buffering: - unsigned char *buffer; - unsigned int buffer_size; - unsigned int buffer_len; - // mpeg block buffer: - unsigned char *b_buffer; - unsigned int b_buffer_size; //size of b_buffer - unsigned int b_buffer_ptr; //index to next data to write - unsigned int b_buffer_len; //len of next data to write - // muxer frame buffer: - unsigned int muxbuf_seen; - // source stream: - void* source; // sh_audio or sh_video - int codec; // codec used for encoding. 0 means copy - // avi stream header: - AVIStreamHeader h; // Rate/Scale and SampleSize must be filled by caller! - // stream specific: - WAVEFORMATEX *wf; - BITMAPINFOHEADER *bih; // in format - int encoder_delay; // in number of frames - int decoder_delay; // in number of frames - int imgfmt; - // mpeg specific: - size_t ipb[3]; // sizes of I/P/B frames - // muxer of that stream - struct muxer_t *muxer; - void *priv; // private stream specific data stored by the muxer - - int vbv_size; - int max_rate; - int avg_rate; -} muxer_stream_t; - -typedef struct { - uint32_t id; - char *text; -} muxer_info_t; - -typedef struct muxer_t{ - // encoding: - MainAVIHeader avih; - off_t movi_start; - off_t movi_end; - off_t file_end; // for MPEG it's system timestamp in 1/90000 s - float audio_delay_fix; - // index: - AVIINDEXENTRY *idx; - int idx_pos; - int idx_size; - // streams: - int num_videos; // for MPEG recalculations - int num_audios; - unsigned int sysrate; // max rate in bytes/s - //int num_streams; - muxer_stream_t* def_v; // default video stream (for general headers) - muxer_stream_t* streams[MUXER_MAX_STREAMS]; - // muxer frame buffer: - struct muxbuf_t * muxbuf; - int muxbuf_num; - int muxbuf_skip_buffer; - // functions: - stream_t *stream; - void (*fix_stream_parameters)(muxer_stream_t *); - void (*cont_write_chunk)(muxer_stream_t *,size_t,unsigned int, double dts, double pts); - void (*cont_write_header)(struct muxer_t *); - void (*cont_write_index)(struct muxer_t *); - muxer_stream_t* (*cont_new_stream)(struct muxer_t *,int); - void *priv; -} muxer_t; - -/* muxer frame buffer */ -typedef struct muxbuf_t { - muxer_stream_t *stream; /* pointer back to corresponding stream */ - double dts; /* decode timestamp / time at which this packet should be feeded into the decoder */ - double pts; /* presentation timestamp / time at which the data in this packet will be presented to the user */ - unsigned char *buffer; - size_t len; - unsigned int flags; -} muxbuf_t; - -extern char *force_fourcc; - -muxer_t *muxer_new_muxer(int type,stream_t *stream); -#define muxer_new_stream(muxer,a) muxer->cont_new_stream(muxer,a) -#define muxer_stream_fix_parameters(muxer, a) muxer->fix_stream_parameters(a) -void muxer_write_chunk(muxer_stream_t *s, size_t len, unsigned int flags, double dts, double pts); -#define muxer_write_header(muxer) muxer->cont_write_header(muxer) -#define muxer_write_index(muxer) muxer->cont_write_index(muxer) - -int muxer_init_muxer_avi(muxer_t *); -int muxer_init_muxer_mpeg(muxer_t *); -int muxer_init_muxer_rawvideo(muxer_t *); -int muxer_init_muxer_lavf(muxer_t *); -int muxer_init_muxer_rawaudio(muxer_t *); - -#endif /* MPLAYER_MUXER_H */ diff --git a/libmpdemux/muxer_avi.c b/libmpdemux/muxer_avi.c deleted file mode 100644 index 8017ef611e..0000000000 --- a/libmpdemux/muxer_avi.c +++ /dev/null @@ -1,694 +0,0 @@ -/* - * This file is part of MPlayer. - * - * MPlayer 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. - * - * MPlayer 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 MPlayer; 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 <inttypes.h> -#include <unistd.h> -#include <limits.h> - -#include "config.h" -#include "mpcommon.h" -#include "stream/stream.h" -#include "demuxer.h" -#include "stheader.h" - -#include "muxer.h" -#include "aviheader.h" -#include "ms_hdr.h" -#include "mp_msg.h" - -extern char *info_name; -extern char *info_artist; -extern char *info_genre; -extern char *info_subject; -extern char *info_copyright; -extern char *info_sourceform; -extern char *info_comment; - -/* #define ODML_CHUNKLEN 0x02000000 */ /* for testing purposes */ -#define ODML_CHUNKLEN 0x40000000 -#define ODML_NOTKEYFRAME 0x80000000U -#define MOVIALIGN 0x00001000 - -float avi_aspect_override = -1.0; -int write_odml = 1; - -struct avi_odmlidx_entry { - uint64_t ofs; - uint32_t len; - uint32_t flags; -}; - -struct avi_odmlsuperidx_entry { - uint64_t ofs; - uint32_t len; - uint32_t duration; -}; - -struct avi_stream_info { - int idxsize; - int idxpos; - int superidxpos; - int superidxsize; - int riffofspos; - int riffofssize; - off_t *riffofs; - struct avi_odmlidx_entry *idx; - struct avi_odmlsuperidx_entry *superidx; -}; - -static unsigned int avi_aspect(muxer_stream_t *vstream) -{ - int x,y; - float aspect = vstream->aspect; - - if (avi_aspect_override > 0.0) { - aspect = avi_aspect_override; - } - - if (aspect <= 0.0) return 0; - - if (aspect > 15.99/9.0 && aspect < 16.01/9.0) { - return MAKE_AVI_ASPECT(16, 9); - } - if (aspect > 3.99/3.0 && aspect < 4.01/3.0) { - return MAKE_AVI_ASPECT(4, 3); - } - - if (aspect >= 1.0) { - x = 16384; - y = (float)x / aspect; - } else { - y = 16384; - x = (float)y * aspect; - } - - return MAKE_AVI_ASPECT(x, y); -} - -static muxer_stream_t* avifile_new_stream(muxer_t *muxer,int type){ - struct avi_stream_info *si; - muxer_stream_t* s; - if (!muxer) return NULL; - if(muxer->avih.dwStreams>=MUXER_MAX_STREAMS){ - mp_msg(MSGT_MUXER, MSGL_ERR, "Too many streams! increase MUXER_MAX_STREAMS !\n"); - return NULL; - } - s=malloc(sizeof(muxer_stream_t)); - memset(s,0,sizeof(muxer_stream_t)); - if(!s) return NULL; // no mem!? - muxer->streams[muxer->avih.dwStreams]=s; - s->type=type; - s->id=muxer->avih.dwStreams; - s->timer=0.0; - s->size=0; - s->muxer=muxer; - s->priv=si=malloc(sizeof(struct avi_stream_info)); - memset(si,0,sizeof(struct avi_stream_info)); - si->idxsize=256; - si->idx=calloc(si->idxsize, sizeof(struct avi_odmlidx_entry)); - si->riffofssize=16; - si->riffofs=calloc((si->riffofssize+1), sizeof(off_t)); - memset(si->riffofs, 0, sizeof(off_t)*si->riffofssize); - - switch(type){ - case MUXER_TYPE_VIDEO: - s->ckid=mmioFOURCC(('0'+s->id/10),('0'+(s->id%10)),'d','c'); - s->h.fccType=streamtypeVIDEO; - if(!muxer->def_v) muxer->def_v=s; - break; - case MUXER_TYPE_AUDIO: - s->ckid=mmioFOURCC(('0'+s->id/10),('0'+(s->id%10)),'w','b'); - s->h.fccType=streamtypeAUDIO; - break; - default: - mp_msg(MSGT_MUXER, MSGL_WARN, "Warning! unknown stream type: %d\n",type); - return NULL; - } - muxer->avih.dwStreams++; - return s; -} - -static void write_avi_chunk(stream_t *stream,unsigned int id,int len,void* data){ - int le_len = le2me_32(len); - int le_id = le2me_32(id); - stream_write_buffer(stream, &le_id, 4); - stream_write_buffer(stream, &le_len, 4); - -if(len>0){ - if(data){ - // DATA - stream_write_buffer(stream, data, len); - if(len&1){ // padding - unsigned char zerobyte=0; - stream_write_buffer(stream, &zerobyte, 1); - } - } else { - // JUNK - char *avi_junk_data="[= MPlayer junk data! =]"; - if(len&1) ++len; // padding - while(len>0){ - int l=strlen(avi_junk_data); - if(l>len) l=len; - stream_write_buffer(stream, avi_junk_data, l); - len-=l; - } - } -} -} - -static void write_avi_list(stream_t *s,unsigned int id,int len); -static void avifile_write_standard_index(muxer_t *muxer); - -static void avifile_odml_new_riff(muxer_t *muxer) -{ - struct avi_stream_info *vsi = muxer->def_v->priv; - uint32_t riff[3]; - - mp_msg(MSGT_MUXER, MSGL_INFO, "ODML: Starting new RIFF chunk at %dMB.\n", (int)(muxer->file_end/1024/1024)); - - vsi->riffofspos++; - if (vsi->riffofspos>=vsi->riffofssize) { - vsi->riffofssize+=16; - vsi->riffofs=realloc_struct(vsi->riffofs,(vsi->riffofssize+1),sizeof(off_t)); - } - vsi->riffofs[vsi->riffofspos] = stream_tell(muxer->stream); - - /* RIFF/AVIX chunk */ - riff[0]=le2me_32(mmioFOURCC('R','I','F','F')); - riff[1]=0; - riff[2]=le2me_32(mmioFOURCC('A','V','I','X')); - stream_write_buffer(muxer->stream, riff, 12); - - write_avi_list(muxer->stream,listtypeAVIMOVIE,0); - - muxer->file_end = stream_tell(muxer->stream); -} - -static void avifile_write_header(muxer_t *muxer); - -static void avifile_write_chunk(muxer_stream_t *s,size_t len,unsigned int flags, double dts, double pts){ - off_t rifflen; - muxer_t *muxer=s->muxer; - struct avi_stream_info *si = s->priv; - struct avi_stream_info *vsi = muxer->def_v->priv; - int paddedlen = len + (len&1); - - if (s->type == MUXER_TYPE_VIDEO && !s->h.dwSuggestedBufferSize) { - off_t pos=stream_tell(muxer->stream); - stream_seek(muxer->stream, 0); - avifile_write_header(muxer); - stream_seek(muxer->stream, pos); - } - if(index_mode){ - rifflen = muxer->file_end - vsi->riffofs[vsi->riffofspos] - 8; - if (vsi->riffofspos == 0) { - rifflen += 8+muxer->idx_pos*sizeof(AVIINDEXENTRY); - } - if (rifflen + paddedlen > ODML_CHUNKLEN && write_odml == 1) { - if (vsi->riffofspos == 0) { - avifile_write_standard_index(muxer); - } - avifile_odml_new_riff(muxer); - } - - if (vsi->riffofspos == 0) { - // add to the traditional index: - if(muxer->idx_pos>=muxer->idx_size){ - muxer->idx_size+=256; // 4kB - muxer->idx=realloc_struct(muxer->idx,muxer->idx_size,16); - } - muxer->idx[muxer->idx_pos].ckid=s->ckid; - muxer->idx[muxer->idx_pos].dwFlags=flags; // keyframe? - muxer->idx[muxer->idx_pos].dwChunkOffset=muxer->file_end-(muxer->movi_start-4); - muxer->idx[muxer->idx_pos].dwChunkLength=len; - ++muxer->idx_pos; - } - - // add to odml index - if(si->idxpos>=si->idxsize){ - si->idxsize+=256; - si->idx=realloc_struct(si->idx,si->idxsize,sizeof(*si->idx)); - } - si->idx[si->idxpos].flags=(flags&AVIIF_KEYFRAME)?0:ODML_NOTKEYFRAME; - si->idx[si->idxpos].ofs=muxer->file_end; - si->idx[si->idxpos].len=len; - ++si->idxpos; - } - // write out the chunk: - write_avi_chunk(muxer->stream,s->ckid,len,s->buffer); /* unsigned char */ - - if (len > s->h.dwSuggestedBufferSize){ - s->h.dwSuggestedBufferSize = len; - } - if((unsigned int)len>s->h.dwSuggestedBufferSize) s->h.dwSuggestedBufferSize=len; - - muxer->file_end += 8 + paddedlen; -} - -static void write_avi_list(stream_t *stream,unsigned int id,int len){ - unsigned int list_id=FOURCC_LIST; - int le_len; - int le_id; - len+=4; // list fix - list_id = le2me_32(list_id); - le_len = le2me_32(len); - le_id = le2me_32(id); - stream_write_buffer(stream, &list_id, 4); - stream_write_buffer(stream, &le_len, 4); - stream_write_buffer(stream, &le_id, 4); -} - -#define WFSIZE(wf) (sizeof(*wf)+(wf)->cbSize) - -static void avifile_write_header(muxer_t *muxer){ - uint32_t riff[3]; - unsigned int dmlh[1]; - unsigned int i; - unsigned int hdrsize; - muxer_info_t info[16]; - VideoPropHeader vprp; - uint32_t aspect = avi_aspect(muxer->def_v); - struct avi_stream_info *vsi = muxer->def_v->priv; - int isodml = vsi->riffofspos > 0; - - mp_tmsg(MSGT_MUXER, MSGL_INFO, "Writing header...\n"); - if (aspect == 0) { - mp_msg(MSGT_MUXER, MSGL_INFO, "ODML: Aspect information not (yet?) available or unspecified, not writing vprp header.\n"); - } else { - mp_msg(MSGT_MUXER, MSGL_INFO, "ODML: vprp aspect is %d:%d.\n", aspect >> 16, aspect & 0xffff); - } - - /* deal with stream delays */ - for (i = 0; muxer->streams[i] && i < MUXER_MAX_STREAMS; ++i) { - muxer_stream_t *s = muxer->streams[i]; - if (s->type == MUXER_TYPE_AUDIO && muxer->audio_delay_fix > 0.0) { - s->h.dwStart = muxer->audio_delay_fix * s->h.dwRate/s->h.dwScale + 0.5; - mp_tmsg(MSGT_MUXER, MSGL_INFO, "Setting audio delay to %5.3fs.\n", (float)s->h.dwStart * s->h.dwScale/s->h.dwRate); - } - if (s->type == MUXER_TYPE_VIDEO && muxer->audio_delay_fix < 0.0) { - s->h.dwStart = -muxer->audio_delay_fix * s->h.dwRate/s->h.dwScale + 0.5; - mp_tmsg(MSGT_MUXER, MSGL_INFO, "Setting video delay to %5.3fs.\n", (float)s->h.dwStart * s->h.dwScale/s->h.dwRate); - } - } - - if (isodml) { - unsigned int rifflen, movilen; - int i; - - vsi->riffofs[vsi->riffofspos+1] = muxer->file_end; - - /* fixup RIFF lengths */ - for (i=0; i<=vsi->riffofspos; i++) { - rifflen = vsi->riffofs[i+1] - vsi->riffofs[i] - 8; - movilen = le2me_32(rifflen - 12); - rifflen = le2me_32(rifflen); - stream_seek(muxer->stream, vsi->riffofs[i]+4); - stream_write_buffer(muxer->stream,&rifflen,4); - - /* fixup movi length */ - if (i > 0) { - stream_seek(muxer->stream, vsi->riffofs[i]+16); - stream_write_buffer(muxer->stream,&movilen,4); - } - } - - stream_seek(muxer->stream, 12); - } else { - // RIFF header: - riff[0]=mmioFOURCC('R','I','F','F'); - riff[1]=muxer->file_end-2*sizeof(unsigned int); // filesize - riff[2]=formtypeAVI; // 'AVI ' - riff[0]=le2me_32(riff[0]); - riff[1]=le2me_32(riff[1]); - riff[2]=le2me_32(riff[2]); - stream_write_buffer(muxer->stream,&riff,12); - } - - // update AVI header: - if(muxer->def_v){ - int i; - muxer->avih.dwMicroSecPerFrame=1000000.0*muxer->def_v->h.dwScale/muxer->def_v->h.dwRate; -// muxer->avih.dwMaxBytesPerSec=1000000; // dummy!!!!! FIXME -// muxer->avih.dwPaddingGranularity=2; // ??? - muxer->avih.dwFlags|=AVIF_ISINTERLEAVED|AVIF_TRUSTCKTYPE; - muxer->avih.dwTotalFrames=0; - for (i=0; i<muxer->idx_pos; i++) { - if (muxer->idx[i].ckid == muxer->def_v->ckid) - muxer->avih.dwTotalFrames++; - } -// muxer->avih.dwSuggestedBufferSize=muxer->def_v->h.dwSuggestedBufferSize; - muxer->avih.dwWidth=muxer->def_v->bih->biWidth; - muxer->avih.dwHeight=muxer->def_v->bih->biHeight; - } - - // AVI header: - hdrsize=sizeof(muxer->avih)+8; - if (isodml) hdrsize+=sizeof(dmlh)+20; // dmlh - // calc total header size: - for(i=0;i<muxer->avih.dwStreams;i++){ - muxer_stream_t *s = muxer->streams[i]; - struct avi_stream_info *si = s->priv; - - hdrsize+=12; // LIST - hdrsize+=sizeof(muxer->streams[i]->h)+8; // strh - switch(muxer->streams[i]->type){ - case MUXER_TYPE_VIDEO: - hdrsize+=muxer->streams[i]->bih->biSize+8; // strf - if (aspect != 0) { - hdrsize+=8+4*(9+8*1); // vprp - } - break; - case MUXER_TYPE_AUDIO: - hdrsize+=WFSIZE(muxer->streams[i]->wf)+8; // strf - break; - } - if (isodml && si && si->superidx && si->superidxsize) { - hdrsize += 32 + 16*si->superidxsize; //indx - } - } - write_avi_list(muxer->stream,listtypeAVIHEADER,hdrsize); - - le2me_MainAVIHeader(&muxer->avih); - write_avi_chunk(muxer->stream,ckidAVIMAINHDR,sizeof(muxer->avih),&muxer->avih); /* MainAVIHeader */ - le2me_MainAVIHeader(&muxer->avih); - - // stream headers: - for(i=0;i<muxer->avih.dwStreams;i++){ - muxer_stream_t *s = muxer->streams[i]; - struct avi_stream_info *si = s->priv; - unsigned int idxhdr[8]; - int j,n; - - hdrsize=sizeof(s->h)+8; // strh - if (si && si->superidx && si->superidxsize) { - hdrsize += 32 + 16*si->superidxsize; //indx - } - switch(s->type){ - case MUXER_TYPE_VIDEO: - hdrsize+=s->bih->biSize+8; // strf - s->h.fccHandler = s->bih->biCompression; - s->h.rcFrame.right = s->bih->biWidth; - s->h.rcFrame.bottom = s->bih->biHeight; - if (aspect != 0) { - // fill out vprp info - memset(&vprp, 0, sizeof(vprp)); - vprp.dwVerticalRefreshRate = (s->h.dwRate+s->h.dwScale-1)/s->h.dwScale; - vprp.dwHTotalInT = muxer->avih.dwWidth; - vprp.dwVTotalInLines = muxer->avih.dwHeight; - vprp.dwFrameAspectRatio = aspect; - vprp.dwFrameWidthInPixels = muxer->avih.dwWidth; - vprp.dwFrameHeightInLines = muxer->avih.dwHeight; - vprp.nbFieldPerFrame = 1; - vprp.FieldInfo[0].CompressedBMHeight = muxer->avih.dwHeight; - vprp.FieldInfo[0].CompressedBMWidth = muxer->avih.dwWidth; - vprp.FieldInfo[0].ValidBMHeight = muxer->avih.dwHeight; - vprp.FieldInfo[0].ValidBMWidth = muxer->avih.dwWidth; - hdrsize+=8+4*(9+8*1); // vprp - } - break; - case MUXER_TYPE_AUDIO: - hdrsize+=WFSIZE(s->wf)+8; // strf - s->h.fccHandler = s->wf->wFormatTag; - break; - } - - write_avi_list(muxer->stream,listtypeSTREAMHEADER,hdrsize); - le2me_AVIStreamHeader(&s->h); - write_avi_chunk(muxer->stream,ckidSTREAMHEADER,sizeof(s->h),&s->h); /* AVISTreamHeader */ // strh - le2me_AVIStreamHeader(&s->h); - - switch(s->type){ - case MUXER_TYPE_VIDEO: -{ - int biSize=s->bih->biSize; - le2me_BITMAPINFOHEADER(s->bih); - write_avi_chunk(muxer->stream,ckidSTREAMFORMAT,biSize,s->bih); /* BITMAPINFOHEADER */ - le2me_BITMAPINFOHEADER(s->bih); - - if (aspect != 0) { - int fields = vprp.nbFieldPerFrame; - le2me_VideoPropHeader(&vprp); - le2me_VIDEO_FIELD_DESC(&vprp.FieldInfo[0]); - le2me_VIDEO_FIELD_DESC(&vprp.FieldInfo[1]); - write_avi_chunk(muxer->stream,mmioFOURCC('v','p','r','p'), - sizeof(VideoPropHeader) - - sizeof(VIDEO_FIELD_DESC)*(2-fields), - &vprp); /* Video Properties Header */ - } -} - break; - case MUXER_TYPE_AUDIO: -{ - int wfsize = WFSIZE(s->wf); - le2me_WAVEFORMATEX(s->wf); - write_avi_chunk(muxer->stream,ckidSTREAMFORMAT,wfsize,s->wf); /* WAVEFORMATEX */ - le2me_WAVEFORMATEX(s->wf); -} - break; - } - if (isodml && si && si->superidx && si->superidxsize) { - n = si->superidxsize; - - idxhdr[0] = le2me_32(mmioFOURCC('i', 'n', 'd', 'x')); - idxhdr[1] = le2me_32(24 + 16*n); - idxhdr[2] = le2me_32(0x00000004); - idxhdr[3] = le2me_32(si->superidxpos); - idxhdr[4] = le2me_32(s->ckid); - idxhdr[5] = 0; - idxhdr[6] = 0; - idxhdr[7] = 0; - - stream_write_buffer(muxer->stream,idxhdr,sizeof(idxhdr)); - for (j=0; j<n; j++) { - struct avi_odmlsuperidx_entry *entry = &si->superidx[j]; - unsigned int data[4]; - data[0] = le2me_32(entry->ofs); - data[1] = le2me_32(entry->ofs >> 32); - data[2] = le2me_32(entry->len); - data[3] = le2me_32(entry->duration); - stream_write_buffer(muxer->stream,data,sizeof(data)); - } - } - } - - // ODML - if (isodml) { - memset(dmlh, 0, sizeof(dmlh)); - dmlh[0] = le2me_32(muxer->avih.dwTotalFrames); - write_avi_list(muxer->stream,mmioFOURCC('o','d','m','l'),sizeof(dmlh)+8); - write_avi_chunk(muxer->stream,mmioFOURCC('d','m','l','h'),sizeof(dmlh),dmlh); - } - -// ============= INFO =============== -// always include software info -info[0].id=mmioFOURCC('I','S','F','T'); // Software: -info[0].text=mencoder_version; -// include any optional strings -i=1; -if(info_name!=NULL){ - info[i].id=mmioFOURCC('I','N','A','M'); // Name: - info[i++].text=info_name; -} -if(info_artist!=NULL){ - info[i].id=mmioFOURCC('I','A','R','T'); // Artist: - info[i++].text=info_artist; -} -if(info_genre!=NULL){ - info[i].id=mmioFOURCC('I','G','N','R'); // Genre: - info[i++].text=info_genre; -} -if(info_subject!=NULL){ - info[i].id=mmioFOURCC('I','S','B','J'); // Subject: - info[i++].text=info_subject; -} -if(info_copyright!=NULL){ - info[i].id=mmioFOURCC('I','C','O','P'); // Copyright: - info[i++].text=info_copyright; -} -if(info_sourceform!=NULL){ - info[i].id=mmioFOURCC('I','S','R','F'); // Source Form: - info[i++].text=info_sourceform; -} -if(info_comment!=NULL){ - info[i].id=mmioFOURCC('I','C','M','T'); // Comment: - info[i++].text=info_comment; -} -info[i].id=0; - - hdrsize=0; - // calc info size: - for(i=0;info[i].id!=0;i++) if(info[i].text){ - size_t sz=strlen(info[i].text)+1; - hdrsize+=sz+8+sz%2; - } - // write infos: - if (hdrsize!=0){ - write_avi_list(muxer->stream,mmioFOURCC('I','N','F','O'),hdrsize); - for(i=0;info[i].id!=0;i++) if(info[i].text){ - write_avi_chunk(muxer->stream,info[i].id,strlen(info[i].text)+1,info[i].text); - } - } - - // JUNK: - write_avi_chunk(muxer->stream,ckidAVIPADDING,MOVIALIGN-(stream_tell(muxer->stream)%MOVIALIGN)-8,NULL); /* junk */ - if (!isodml) { - // 'movi' header: - write_avi_list(muxer->stream,listtypeAVIMOVIE,muxer->movi_end-stream_tell(muxer->stream)-12); - } else { - if (stream_tell(muxer->stream) != MOVIALIGN) { - mp_msg(MSGT_MUXER, MSGL_ERR, "Opendml superindex is too big for reserved space!\n"); - mp_msg(MSGT_MUXER, MSGL_ERR, "Expected filepos %d, real filepos %ld, missing space %ld\n", MOVIALIGN, stream_tell(muxer->stream), stream_tell(muxer->stream)-MOVIALIGN); - mp_msg(MSGT_MUXER, MSGL_ERR, "Try increasing MOVIALIGN in libmpdemux/muxer_avi.c\n"); - } - write_avi_list(muxer->stream,listtypeAVIMOVIE,muxer->movi_end-stream_tell(muxer->stream)-12); - } - muxer->movi_start=stream_tell(muxer->stream); - if (muxer->file_end == 0) muxer->file_end = stream_tell(muxer->stream); -} - -static void avifile_odml_write_index(muxer_t *muxer){ - muxer_stream_t* s; - struct avi_stream_info *si; - int i; - - for (i=0; i<muxer->avih.dwStreams; i++) { - int j,k,n,idxpos,len,last,entries_per_subidx; - unsigned int idxhdr[8]; - s = muxer->streams[i]; - si = s->priv; - - /* - * According to Avery Lee MSMP wants the subidx chunks to have the same size. - * - * So this code figures out how many entries we can put into - * an ix?? chunk, so that each ix?? chunk has the same size and the offsets - * don't overflow (Using ODML_CHUNKLEN for that is a bit more restrictive - * than it has to be though). - */ - - len = 0; - n = 0; - entries_per_subidx = INT_MAX; - do { - off_t start = si->idx[0].ofs; - last = entries_per_subidx; - for (j=0; j<si->idxpos; j++) { - len = si->idx[j].ofs - start; - if(len >= ODML_CHUNKLEN || n >= entries_per_subidx) { - if (entries_per_subidx > n) { - entries_per_subidx = n; - } - start = si->idx[j].ofs; - len = 0; - n = 0; - } - n++; - } - } while (last != entries_per_subidx); - - si->superidxpos = (si->idxpos+entries_per_subidx-1) / entries_per_subidx; - - mp_msg(MSGT_MUXER, MSGL_V, "ODML: Stream %d: Using %d entries per subidx, %d entries in superidx\n", - i, entries_per_subidx, si->superidxpos); - - si->superidxsize = si->superidxpos; - si->superidx = calloc(si->superidxsize, sizeof(*si->superidx)); - memset(si->superidx, 0, sizeof(*si->superidx) * si->superidxsize); - - idxpos = 0; - for (j=0; j<si->superidxpos; j++) { - off_t start = si->idx[idxpos].ofs; - int duration; - - duration = 0; - for (k=0; k<entries_per_subidx && idxpos+k<si->idxpos; k++) { - duration += s->h.dwSampleSize ? si->idx[idxpos+k].len/s->h.dwSampleSize : 1; - } - - idxhdr[0] = le2me_32((s->ckid << 16) | mmioFOURCC('i', 'x', 0, 0)); - idxhdr[1] = le2me_32(24 + 8*k); - idxhdr[2] = le2me_32(0x01000002); - idxhdr[3] = le2me_32(k); - idxhdr[4] = le2me_32(s->ckid); - idxhdr[5] = le2me_32(start + 8); - idxhdr[6] = le2me_32((start + 8)>> 32); - idxhdr[7] = 0; /* unused */ - - si->superidx[j].len = 32 + 8*k; - si->superidx[j].ofs = stream_tell(muxer->stream); - si->superidx[j].duration = duration; - - stream_write_buffer(muxer->stream, idxhdr,sizeof(idxhdr)); - for (k=0; k<entries_per_subidx && idxpos<si->idxpos; k++) { - unsigned int entry[2]; - entry[0] = le2me_32(si->idx[idxpos].ofs - start); - entry[1] = le2me_32(si->idx[idxpos].len | si->idx[idxpos].flags); - idxpos++; - stream_write_buffer(muxer->stream, entry, sizeof(entry)); - } - } - } - muxer->file_end=stream_tell(muxer->stream); -} - -static void avifile_write_standard_index(muxer_t *muxer){ - - muxer->movi_end=stream_tell(muxer->stream); - if(muxer->idx && muxer->idx_pos>0){ - int i; - // fixup index entries: -// for(i=0;i<muxer->idx_pos;i++) muxer->idx[i].dwChunkOffset-=muxer->movi_start-4; - // write index chunk: - for (i=0; i<muxer->idx_pos; i++) le2me_AVIINDEXENTRY((&muxer->idx[i])); - write_avi_chunk(muxer->stream,ckidAVINEWINDEX,16*muxer->idx_pos,muxer->idx); /* AVIINDEXENTRY */ - for (i=0; i<muxer->idx_pos; i++) le2me_AVIINDEXENTRY((&muxer->idx[i])); - muxer->avih.dwFlags|=AVIF_HASINDEX; - } - muxer->file_end=stream_tell(muxer->stream); -} - -static void avifile_write_index(muxer_t *muxer){ - struct avi_stream_info *vsi |