⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 muxer_avi.c

📁 自己移植的linux下的流媒体播放器原代码,支持mms协议,支持ftp和http协议.
💻 C
📖 第 1 页 / 共 2 页
字号:
#include <stdio.h>#include <stdlib.h>#include <string.h>#include <inttypes.h>#include <unistd.h>#include <limits.h>#include "config.h"#include "../version.h"#include "stream.h"#include "demuxer.h"#include "stheader.h"#include "bswap.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        0x00001000float 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=malloc(sizeof(struct avi_odmlidx_entry)*si->idxsize);    si->riffofssize=16;    si->riffofs=malloc(sizeof(off_t)*(si->riffofssize+1));    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(FILE *f,unsigned int id,int len,void* data){ int le_len = le2me_32(len); int le_id = le2me_32(id); fwrite(&le_id,4,1,f); fwrite(&le_len,4,1,f);if(len>0){  if(data){    // DATA    fwrite(data,len,1,f);    if(len&1){  // padding      unsigned char zerobyte=0;      fwrite(&zerobyte,1,1,f);    }  } 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;      fwrite(avi_junk_data,l,1,f);      len-=l;    }  }}}static void write_avi_list(FILE *f,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;    FILE *f = muxer->file;    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(vsi->riffofs,sizeof(off_t)*(vsi->riffofssize+1));    }    vsi->riffofs[vsi->riffofspos] = ftello(f);    /* RIFF/AVIX chunk */    riff[0]=le2me_32(mmioFOURCC('R','I','F','F'));    riff[1]=0;    riff[2]=le2me_32(mmioFOURCC('A','V','I','X'));    fwrite(riff,12,1,f);    write_avi_list(f,listtypeAVIMOVIE,0);    muxer->file_end = ftello(f);}static void avifile_write_header(muxer_t *muxer);static void avifile_write_chunk(muxer_stream_t *s,size_t len,unsigned int flags){    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=ftell(muxer->file);	fseek(muxer->file, 0, SEEK_SET);	avifile_write_header(muxer);	fseek(muxer->file, pos, SEEK_SET);    }    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(muxer->idx,16*muxer->idx_size);        }        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(si->idx,sizeof(*si->idx)*si->idxsize);    }    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->file,s->ckid,len,s->buffer); /* unsigned char */    // alter counters:    if (len > s->h.dwSuggestedBufferSize){	s->h.dwSuggestedBufferSize = len;    }    if(s->h.dwSampleSize){	// CBR	s->h.dwLength+=len/s->h.dwSampleSize;	if(len%s->h.dwSampleSize) mp_msg(MSGT_MUXER, MSGL_WARN, "Warning! len isn't divisable by samplesize!\n");    } else {	// VBR	s->h.dwLength++;    }    s->timer=(double)s->h.dwLength*s->h.dwScale/s->h.dwRate;    s->size+=len;    if((unsigned int)len>s->h.dwSuggestedBufferSize) s->h.dwSuggestedBufferSize=len;    muxer->file_end += 8 + paddedlen;}static void write_avi_list(FILE *f,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);  fwrite(&list_id,4,1,f);  fwrite(&le_len,4,1,f);  fwrite(&le_id,4,1,f);}#define WFSIZE(wf) (sizeof(WAVEFORMATEX)+(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];  FILE *f = muxer->file;  VideoPropHeader vprp;  uint32_t aspect = avi_aspect(muxer->def_v);  off_t pos;  struct avi_stream_info *vsi = muxer->def_v->priv;  int isodml = vsi->riffofspos > 0;  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);  }  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);          fseeko(f, vsi->riffofs[i]+4, SEEK_SET);          fwrite(&rifflen,4,1,f);          /* fixup movi length */          if (i > 0) {              fseeko(f, vsi->riffofs[i]+16, SEEK_SET);              fwrite(&movilen,4,1,f);          }      }      fseeko(f, 12, SEEK_SET);  } 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]);    fwrite(&riff,12,1,f);  }  // 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; // ???

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -