demux_avs.c

来自「君正早期ucos系统(只有早期的才不没有打包成库),MPLAYER,文件系统,图」· C语言 代码 · 共 480 行

C
480
字号
/* * Demuxer for avisynth * Copyright (c) 2005 Gianluigi Tiesi <sherpya@netfarm.it> * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library 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 * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library; if not, write to the * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301 USA. */#include <mplaylib.h>#include <mplaylib.h>#include <mplaylib.h>#include "config.h"#include "mp_msg.h"#include "help_mp.h"#include "stream/stream.h"#include "demuxer.h"#include "stheader.h"#include "libvo/fastmemcpy.h"#include "loader/wine/windef.h"#ifdef WIN32_LOADER#include "loader/ldt_keeper.h"#endif#include "demux_avs.h"#define MAX_AVS_SIZE    16 * 1024 /* 16k should be enough */HMODULE WINAPI LoadLibraryA(LPCSTR);FARPROC WINAPI GetProcAddress(HMODULE,LPCSTR);int     WINAPI FreeLibrary(HMODULE);typedef WINAPI AVS_ScriptEnvironment* (*imp_avs_create_script_environment)(int version);typedef WINAPI AVS_Value (*imp_avs_invoke)(AVS_ScriptEnvironment *, const char * name, AVS_Value args, const char** arg_names);typedef WINAPI const AVS_VideoInfo *(*imp_avs_get_video_info)(AVS_Clip *);typedef WINAPI AVS_Clip* (*imp_avs_take_clip)(AVS_Value, AVS_ScriptEnvironment *);typedef WINAPI void (*imp_avs_release_clip)(AVS_Clip *);typedef WINAPI AVS_VideoFrame* (*imp_avs_get_frame)(AVS_Clip *, int n);typedef WINAPI void (*imp_avs_release_video_frame)(AVS_VideoFrame *);typedef WINAPI int (*imp_avs_get_audio)(AVS_Clip *, void * buf, uint64_t start, uint64_t count); #define Q(string) # string#define IMPORT_FUNC(x) \    AVS->x = ( imp_##x ) GetProcAddress(AVS->dll, Q(x)); \    if (!AVS->x) { mp_msg(MSGT_DEMUX,MSGL_V,"AVS: failed to load "Q(x)"()\n"); return 0; }typedef struct tagAVS{    AVS_ScriptEnvironment *avs_env;    AVS_Value handler;    AVS_Clip *clip;    const AVS_VideoInfo *video_info;#ifdef WIN32_LOADER    ldt_fs_t* ldt_fs;#endif    HMODULE dll;    int frameno;    uint64_t sampleno;    int init;        imp_avs_create_script_environment avs_create_script_environment;    imp_avs_invoke avs_invoke;    imp_avs_get_video_info avs_get_video_info;    imp_avs_take_clip avs_take_clip;    imp_avs_release_clip avs_release_clip;    imp_avs_get_frame avs_get_frame;    imp_avs_release_video_frame avs_release_video_frame;    imp_avs_get_audio avs_get_audio;} AVS_T;AVS_T *initAVS(const char *filename){       AVS_T *AVS = malloc (sizeof(AVS_T));    AVS_Value arg0 = avs_new_value_string(filename);    AVS_Value args = avs_new_value_array(&arg0, 1);        memset(AVS, 0, sizeof(AVS_T));#ifdef WIN32_LOADER    AVS->ldt_fs = Setup_LDT_Keeper();#endif        AVS->dll = LoadLibraryA("avisynth.dll");    if(!AVS->dll)    {        mp_msg(MSGT_DEMUX ,MSGL_V, "AVS: failed to load avisynth.dll\n");        goto avs_err;    }        /* Dynamic import of needed stuff from avisynth.dll */    IMPORT_FUNC(avs_create_script_environment);    IMPORT_FUNC(avs_invoke);    IMPORT_FUNC(avs_get_video_info);    IMPORT_FUNC(avs_take_clip);    IMPORT_FUNC(avs_release_clip);    IMPORT_FUNC(avs_get_frame);    IMPORT_FUNC(avs_release_video_frame);    IMPORT_FUNC(avs_get_audio);        AVS->avs_env = AVS->avs_create_script_environment(AVISYNTH_INTERFACE_VERSION);    if (!AVS->avs_env)    {        mp_msg(MSGT_DEMUX, MSGL_V, "AVS: avs_create_script_environment failed\n");        goto avs_err;    }        AVS->handler = AVS->avs_invoke(AVS->avs_env, "Import", args, 0);        if (avs_is_error(AVS->handler))    {        mp_msg(MSGT_DEMUX, MSGL_V, "AVS: Avisynth error: %s\n", avs_as_string(AVS->handler));        goto avs_err;    }    if (!avs_is_clip(AVS->handler))    {        mp_msg(MSGT_DEMUX, MSGL_V, "AVS: Avisynth doesn't return a clip\n");        goto avs_err;    }        return AVS;avs_err:    if (AVS->dll) FreeLibrary(AVS->dll);#ifdef WIN32_LOADER    Restore_LDT_Keeper(AVS->ldt_fs);#endif    free(AVS);    return NULL;}/* Implement RGB MODES ?? */#if 0static __inline int get_mmioFOURCC(const AVS_VideoInfo *v){    if (avs_is_rgb(v)) return mmioFOURCC(8, 'R', 'G', 'B');    if (avs_is_rgb24(v)) return mmioFOURCC(24, 'R', 'G', 'B');    if (avs_is_rgb32(v)) return mmioFOURCC(32, 'R', 'G', 'B');    if (avs_is_yv12(v)) return mmioFOURCC('Y', 'V', '1', '2');    if (avs_is_yuy(v)) return mmioFOURCC('Y', 'U', 'Y', ' ');    if (avs_is_yuy2(v)) return mmioFOURCC('Y', 'U', 'Y', '2');        return 0;}#endifstatic int demux_avs_fill_buffer(demuxer_t *demuxer, demux_stream_t *ds){    AVS_VideoFrame *curr_frame;    demux_packet_t *dp = NULL;    AVS_T *AVS = demuxer->priv;    if (ds == demuxer->video)    {        sh_video_t *sh_video = demuxer->video->sh;        char *dst;        int w, h;        if (AVS->video_info->num_frames <= AVS->frameno) return 0; // EOF        curr_frame = AVS->avs_get_frame(AVS->clip, AVS->frameno);        if (!curr_frame)        {            mp_msg(MSGT_DEMUX, MSGL_V, "AVS: error getting frame -- EOF??\n");            return 0;        }        w = curr_frame->row_size;        h = curr_frame->height;        dp = new_demux_packet(w * h + 2 * (w / 2) * (h / 2));        dp->pts=AVS->frameno / sh_video->fps;        dst = dp->buffer;        memcpy_pic(dst, curr_frame->vfb->data + curr_frame->offset,                   w, h, w, curr_frame->pitch);        dst += w * h;        w /= 2; h /= 2;        memcpy_pic(dst, curr_frame->vfb->data + curr_frame->offsetV,                   w, h, w, curr_frame->pitchUV);        dst += w * h;        memcpy_pic(dst, curr_frame->vfb->data + curr_frame->offsetU,                   w, h, w, curr_frame->pitchUV);        ds_add_packet(demuxer->video, dp);        AVS->frameno++;        AVS->avs_release_video_frame(curr_frame);    }        /* Audio */    if (ds == demuxer->audio)    {        sh_audio_t *sh_audio = ds->sh;        int samples = sh_audio->samplerate;        uint64_t l;        samples = FFMIN(samples, AVS->video_info->num_audio_samples - AVS->sampleno);        if (!samples) return 0;        l = samples * sh_audio->channels * sh_audio->samplesize;        if (l > INT_MAX) {            mp_msg(MSGT_DEMUX, MSGL_FATAL, "AVS: audio packet too big\n");            return 0;        }        dp = new_demux_packet(l);        dp->pts = AVS->sampleno / sh_audio->samplerate;                if (AVS->avs_get_audio(AVS->clip, dp->buffer, AVS->sampleno, samples))        {            mp_msg(MSGT_DEMUX, MSGL_V, "AVS: avs_get_audio() failed\n");            return 0;        }        ds_add_packet(demuxer->audio, dp);        AVS->sampleno += samples;    }        return 1;}static demuxer_t* demux_open_avs(demuxer_t* demuxer){    int found = 0;    AVS_T *AVS = demuxer->priv;    int audio_samplesize = 0;    AVS->frameno = 0;    AVS->sampleno = 0;    mp_msg(MSGT_DEMUX, MSGL_V, "AVS: demux_open_avs()\n");    demuxer->seekable = 1;    AVS->clip = AVS->avs_take_clip(AVS->handler, AVS->avs_env);    if(!AVS->clip)    {        mp_msg(MSGT_DEMUX, MSGL_V, "AVS: avs_take_clip() failed\n");        return NULL;    }    AVS->video_info = AVS->avs_get_video_info(AVS->clip);    if (!AVS->video_info)    {        mp_msg(MSGT_DEMUX, MSGL_V, "AVS: avs_get_video_info() call failed\n");        return NULL;    }        if (!avs_is_yv12(AVS->video_info))    {        AVS->handler = AVS->avs_invoke(AVS->avs_env, "ConvertToYV12", avs_new_value_array(&AVS->handler, 1), 0);        if (avs_is_error(AVS->handler))        {            mp_msg(MSGT_DEMUX, MSGL_V, "AVS: Cannot convert input video to YV12: %s\n", avs_as_string(AVS->handler));            return NULL;        }                AVS->clip = AVS->avs_take_clip(AVS->handler, AVS->avs_env);                if(!AVS->clip)        {            mp_msg(MSGT_DEMUX, MSGL_V, "AVS: avs_take_clip() failed\n");            return NULL;        }        AVS->video_info = AVS->avs_get_video_info(AVS->clip);        if (!AVS->video_info)        {            mp_msg(MSGT_DEMUX, MSGL_V, "AVS: avs_get_video_info() call failed\n");            return NULL;        }    }        // TODO check field-based ??    /* Video */      if (avs_has_video(AVS->video_info))    {        sh_video_t *sh_video = new_sh_video(demuxer, 0);        found = 1;                if (demuxer->video->id == -1) demuxer->video->id = 0;        if (demuxer->video->id == 0)        demuxer->video->sh = sh_video;        sh_video->ds = demuxer->video;                sh_video->disp_w = AVS->video_info->width;        sh_video->disp_h = AVS->video_info->height;                //sh_video->format = get_mmioFOURCC(AVS->video_info);        sh_video->format = mmioFOURCC('Y', 'V', '1', '2');        sh_video->fps = (float) ((float) AVS->video_info->fps_numerator / (float) AVS->video_info->fps_denominator);        sh_video->frametime = 1.0 / sh_video->fps;                sh_video->bih = malloc(sizeof(BITMAPINFOHEADER) + (256 * 4));        sh_video->bih->biCompression = sh_video->format;        sh_video->bih->biBitCount = avs_bits_per_pixel(AVS->video_info);        //sh_video->bih->biPlanes = 2;                sh_video->bih->biWidth = AVS->video_info->width;        sh_video->bih->biHeight = AVS->video_info->height;        sh_video->num_frames = 0;        sh_video->num_frames_decoded = 0;    }        /* Audio */    if (avs_has_audio(AVS->video_info))      switch (AVS->video_info->sample_type) {        case AVS_SAMPLE_INT8:  audio_samplesize = 1; break;        case AVS_SAMPLE_INT16: audio_samplesize = 2; break;        case AVS_SAMPLE_INT24: audio_samplesize = 3; break;        case AVS_SAMPLE_INT32:        case AVS_SAMPLE_FLOAT: audio_samplesize = 4; break;        default:          mp_msg(MSGT_DEMUX, MSGL_ERR, "AVS: unknown audio type, disabling\n");      }    if (audio_samplesize)    {        sh_audio_t *sh_audio = new_sh_audio(demuxer, 0);        found = 1;        mp_msg(MSGT_DEMUX, MSGL_V, "AVS: Clip has audio -> Channels = %d - Freq = %d\n", AVS->video_info->nchannels, AVS->video_info->audio_samples_per_second);        if (demuxer->audio->id == -1) demuxer->audio->id = 0;        if (demuxer->audio->id == 0)        demuxer->audio->sh = sh_audio;        sh_audio->ds = demuxer->audio;                sh_audio->wf = malloc(sizeof(WAVEFORMATEX));        sh_audio->wf->wFormatTag = sh_audio->format =            (AVS->video_info->sample_type == AVS_SAMPLE_FLOAT) ? 0x3 : 0x1;        sh_audio->wf->nChannels = sh_audio->channels = AVS->video_info->nchannels;        sh_audio->wf->nSamplesPerSec = sh_audio->samplerate = AVS->video_info->audio_samples_per_second;        sh_audio->samplesize = audio_samplesize;        sh_audio->wf->nAvgBytesPerSec = sh_audio->channels * sh_audio->samplesize * sh_audio->samplerate;        sh_audio->wf->nBlockAlign = sh_audio->channels * sh_audio->samplesize;        sh_audio->wf->wBitsPerSample = sh_audio->samplesize * 8;        sh_audio->wf->cbSize = 0;        sh_audio->i_bps = sh_audio->wf->nAvgBytesPerSec;    }    AVS->init = 1;    if (found)        return demuxer;    else        return NULL;}static int demux_avs_control(demuxer_t *demuxer, int cmd, void *arg){       sh_video_t *sh_video=demuxer->video->sh;    sh_audio_t *sh_audio=demuxer->audio->sh;    AVS_T *AVS = demuxer->priv;    switch(cmd)    {        case DEMUXER_CTRL_GET_TIME_LENGTH:        {            double res = sh_video ? (double)AVS->video_info->num_frames / sh_video->fps : 0;            if (sh_audio)              res = FFMAX(res, (double)AVS->video_info->num_audio_samples / sh_audio->samplerate);            *((double *)arg) = res;            return DEMUXER_CTRL_OK;        }        case DEMUXER_CTRL_GET_PERCENT_POS:        {            if (sh_video)            *((int *)arg) = AVS->frameno * 100 / AVS->video_info->num_frames;            else              *((int *)arg) = AVS->sampleno * 100 / AVS->video_info->num_audio_samples;            return DEMUXER_CTRL_OK;        }    default:        return DEMUXER_CTRL_NOTIMPL;    }}static void demux_close_avs(demuxer_t* demuxer){    AVS_T *AVS = demuxer->priv;    if (AVS)    {        if (AVS->dll)        {            if (AVS->clip)                AVS->avs_release_clip(AVS->clip);            mp_msg(MSGT_DEMUX, MSGL_V, "AVS: Unloading avisynth.dll\n");            FreeLibrary(AVS->dll);        }#ifdef WIN32_LOADER        Restore_LDT_Keeper(AVS->ldt_fs);#endif        free(AVS);    }}static void demux_seek_avs(demuxer_t *demuxer, float rel_seek_secs, float audio_delay, int flags){    sh_video_t *sh_video=demuxer->video->sh;    sh_audio_t *sh_audio=demuxer->audio->sh;    AVS_T *AVS = demuxer->priv;    double video_pos = sh_video ?                       (double)AVS->frameno / sh_video->fps :                       (double)AVS->sampleno / sh_audio->samplerate;    double duration = sh_video ?                      (double)AVS->video_info->num_frames / sh_video->fps :                      (double)AVS->video_info->num_audio_samples / sh_audio->samplerate;        //mp_msg(MSGT_DEMUX, MSGL_V, "AVS: seek rel_seek_secs = %f - flags = %x\n", rel_seek_secs, flags);        // seek absolute    if (flags&1) video_pos=0;    // seek precent    if (flags&2) rel_seek_secs *= duration;    video_pos += rel_seek_secs;    if (video_pos < 0) video_pos = 0;            if (sh_video) {      AVS->frameno = FFMIN(video_pos * sh_video->fps,                           AVS->video_info->num_frames);      sh_video->num_frames_decoded = AVS->frameno;      sh_video->num_frames = AVS->frameno;    }    if (sh_audio)      AVS->sampleno = FFMIN(video_pos * sh_audio->samplerate,                            AVS->video_info->num_audio_samples);}static int avs_check_file(demuxer_t *demuxer){    mp_msg(MSGT_DEMUX, MSGL_V, "AVS: avs_check_file - attempting to open file %s\n", demuxer->filename);    if (!demuxer->filename) return 0;        /* Avoid crazy memory eating when passing an mpg stream */    if (demuxer->movi_end > MAX_AVS_SIZE)    {        mp_msg(MSGT_DEMUX,MSGL_V, "AVS: File is too big, aborting...\n");        return 0;    }        demuxer->priv = initAVS(demuxer->filename);        if (demuxer->priv)    {        mp_msg(MSGT_DEMUX,MSGL_V, "AVS: Init Ok\n");        return DEMUXER_TYPE_AVS;    }    mp_msg(MSGT_DEMUX,MSGL_V, "AVS: Init failed\n");    return 0;}demuxer_desc_t demuxer_desc_avs = {  "Avisynth demuxer",  "avs",  "AVS",  "Gianluigi Tiesi",  "Requires binary dll",  DEMUXER_TYPE_AVS,  0, // unsafe autodetect  avs_check_file,  demux_avs_fill_buffer,  demux_open_avs,  demux_close_avs,  demux_seek_avs,  demux_avs_control};

⌨️ 快捷键说明

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