opt.c

来自「播放H264文件的播放器」· C语言 代码 · 共 471 行 · 第 1/2 页

C
471
字号
/* * AVOptions * Copyright (c) 2005 Michael Niedermayer <michaelni@gmx.at> * * This file is part of FFmpeg. * * FFmpeg is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * FFmpeg 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with FFmpeg; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA *//** * @file libavcodec/opt.c * AVOptions * @author Michael Niedermayer <michaelni@gmx.at> */#include "avcodec.h"#include "opt.h"#include "eval.h"//FIXME order them and do a bin searchconst AVOption *av_find_opt(void *v, const char *name, const char *unit, int mask, int flags){    AVClass *c= *(AVClass**)v; //FIXME silly way of storing AVClass    const AVOption *o= c->option;    for(;o && o->name; o++){        if(!strcmp(o->name, name) && (!unit || (o->unit && !strcmp(o->unit, unit))) && (o->flags & mask) == flags )            return o;    }    return NULL;}const AVOption *av_next_option(void *obj, const AVOption *last){    if(last && last[1].name) return ++last;    else if(last)            return NULL;    else                     return (*(AVClass**)obj)->option;}static int av_set_number2(void *obj, const char *name, double num, int den, int64_t intnum, const AVOption **o_out){    const AVOption *o= av_find_opt(obj, name, NULL, 0, 0);    void *dst;    if(o_out)        *o_out= o;    if(!o || o->offset<=0)        return AVERROR(ENOENT);    if(o->max*den < num*intnum || o->min*den > num*intnum) {        av_log(NULL, AV_LOG_ERROR, "Value %lf for parameter '%s' out of range\n", num, name);        return AVERROR(ERANGE);    }    dst= ((uint8_t*)obj) + o->offset;    switch(o->type){    case FF_OPT_TYPE_FLAGS:    case FF_OPT_TYPE_INT:   *(int       *)dst= llrint(num/den)*intnum; break;    case FF_OPT_TYPE_INT64: *(int64_t   *)dst= llrint(num/den)*intnum; break;    case FF_OPT_TYPE_FLOAT: *(float     *)dst= num*intnum/den;         break;    case FF_OPT_TYPE_DOUBLE:*(double    *)dst= num*intnum/den;         break;    case FF_OPT_TYPE_RATIONAL:        if((int)num == num) 		{			//*(AVRational*)dst= (AVRational){num*intnum, den};			((AVRational*)dst)->num = num*intnum;			((AVRational*)dst)->den = den;		}        else		{			//*(AVRational*)dst= av_d2q(num*intnum/den, 1<<24);			((AVRational*)dst)->num = num*intnum/den;			((AVRational*)dst)->den = 1<<24;		}        break;    default:        return AVERROR(EINVAL);    }    return 0;}static const AVOption *av_set_number(void *obj, const char *name, double num, int den, int64_t intnum){    const AVOption *o = NULL;    if (av_set_number2(obj, name, num, den, intnum, &o) < 0)        return NULL;    else        return o;}static const double const_values[]={    M_PI,    M_E,    FF_QP2LAMBDA,    0};static const char * const const_names[]={    "PI",    "E",    "QP2LAMBDA",    0};static int hexchar2int(char c) {    if (c >= '0' && c <= '9') return c - '0';    if (c >= 'a' && c <= 'f') return c - 'a' + 10;    if (c >= 'A' && c <= 'F') return c - 'A' + 10;    return -1;}int av_set_string3(void *obj, const char *name, const char *val, int alloc, const AVOption **o_out){    int ret;    const AVOption *o= av_find_opt(obj, name, NULL, 0, 0);    if (o_out)        *o_out = o;    if(!o)        return AVERROR(ENOENT);    if(!val || o->offset<=0)        return AVERROR(EINVAL);    if(o->type == FF_OPT_TYPE_BINARY){        uint8_t **dst = (uint8_t **)(((uint8_t*)obj) + o->offset);        int *lendst = (int *)(dst + 1);        uint8_t *bin, *ptr;        int len = strlen(val);        av_freep(dst);        *lendst = 0;        if (len & 1) return AVERROR(EINVAL);        len /= 2;        ptr = bin = av_malloc(len);        while (*val) {            int a = hexchar2int(*val++);            int b = hexchar2int(*val++);            if (a < 0 || b < 0) {                av_free(bin);                return AVERROR(EINVAL);            }            *ptr++ = (a << 4) | b;        }        *dst = bin;        *lendst = len;        return 0;    }    if(o->type != FF_OPT_TYPE_STRING){        int notfirst=0;        for(;;){            int i;            char buf[256];            int cmd=0;            double d;            const char *error = NULL;            if(*val == '+' || *val == '-')                cmd= *(val++);            for(i=0; i<sizeof(buf)-1 && val[i] && val[i]!='+' && val[i]!='-'; i++)                buf[i]= val[i];            buf[i]=0;            /*d = ff_eval2(buf, const_values, const_names, NULL, NULL, NULL, NULL, NULL, &error);            if(isnan(d)) {                const AVOption *o_named= av_find_opt(obj, buf, o->unit, 0, 0);                if(o_named && o_named->type == FF_OPT_TYPE_CONST)                    d= o_named->default_val;                else if(!strcmp(buf, "default")) d= o->default_val;                else if(!strcmp(buf, "max"    )) d= o->max;                else if(!strcmp(buf, "min"    )) d= o->min;                else if(!strcmp(buf, "none"   )) d= 0;                else if(!strcmp(buf, "all"    )) d= ~0;                else {                    if (error)                        av_log(NULL, AV_LOG_ERROR, "Unable to parse option value \"%s\": %s\n", val, error);                    return AVERROR(EINVAL);                }            }*/            if(o->type == FF_OPT_TYPE_FLAGS){                if     (cmd=='+') d= av_get_int(obj, name, NULL) | (int64_t)d;                else if(cmd=='-') d= av_get_int(obj, name, NULL) &~(int64_t)d;            }else{                if     (cmd=='+') d= notfirst*av_get_double(obj, name, NULL) + d;                else if(cmd=='-') d= notfirst*av_get_double(obj, name, NULL) - d;            }            if ((ret = av_set_number2(obj, name, d, 1, 1, o_out)) < 0)                return ret;            val+= i;            if(!*val)                return 0;            notfirst=1;        }        return AVERROR(EINVAL);    }    if(alloc){        av_free(*(void**)(((uint8_t*)obj) + o->offset));        val= av_strdup(val);    }    memcpy(((uint8_t*)obj) + o->offset, &val, sizeof(val));    return 0;}#if LIBAVCODEC_VERSION_MAJOR < 53const AVOption *av_set_string2(void *obj, const char *name, const char *val, int alloc){    const AVOption *o;    if (av_set_string3(obj, name, val, alloc, &o) < 0)        return NULL;    return o;}const AVOption *av_set_string(void *obj, const char *name, const char *val){    const AVOption *o;    if (av_set_string3(obj, name, val, 0, &o) < 0)        return NULL;    return o;}#endifconst AVOption *av_set_double(void *obj, const char *name, double n){    return av_set_number(obj, name, n, 1, 1);}const AVOption *av_set_q(void *obj, const char *name, AVRational n){    return av_set_number(obj, name, n.num, n.den, 1);}const AVOption *av_set_int(void *obj, const char *name, int64_t n){

⌨️ 快捷键说明

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