diff --git a/libavutil/opt.c b/libavutil/opt.c index 2e270cd304..06d3056a37 100644 --- a/libavutil/opt.c +++ b/libavutil/opt.c @@ -1799,6 +1799,51 @@ void *av_opt_ptr(const AVClass *class, void *obj, const char *name) return (uint8_t*)obj + opt->offset; } +static int opt_copy_elem(void *logctx, enum AVOptionType type, + void *dst, const void *src) +{ + uint8_t **dst8 = (uint8_t **)dst; + const uint8_t **src8 = (const uint8_t **)src; + + if (type == AV_OPT_TYPE_STRING) { + if (*dst8 != *src8) + av_freep(dst8); + *dst8 = av_strdup(*src8); + if (*src8 && !*dst8) + return AVERROR(ENOMEM); + } else if (type == AV_OPT_TYPE_BINARY) { + int len = *(const int *)(src8 + 1); + if (*dst8 != *src8) + av_freep(dst8); + *dst8 = av_memdup(*src8, len); + if (len && !*dst8) { + *(int *)(dst8 + 1) = 0; + return AVERROR(ENOMEM); + } + *(int *)(dst8 + 1) = len; + } else if (type == AV_OPT_TYPE_CONST) { + // do nothing + } else if (type == AV_OPT_TYPE_DICT) { + AVDictionary **sdict = (AVDictionary **)src; + AVDictionary **ddict = (AVDictionary **)dst; + if (*sdict != *ddict) + av_dict_free(ddict); + *ddict = NULL; + return av_dict_copy(ddict, *sdict, 0); + } else if (type == AV_OPT_TYPE_CHLAYOUT) { + if (dst != src) + return av_channel_layout_copy(dst, src); + } else if (opt_is_pod(type)) { + size_t size = opt_elem_size[type]; + memcpy(dst, src, size); + } else { + av_log(logctx, AV_LOG_ERROR, "Unhandled option type: %d\n", type); + return AVERROR(EINVAL); + } + + return 0; +} + int av_opt_copy(void *dst, const void *src) { const AVOption *o = NULL; @@ -1815,48 +1860,10 @@ int av_opt_copy(void *dst, const void *src) while ((o = av_opt_next(src, o))) { void *field_dst = (uint8_t *)dst + o->offset; void *field_src = (uint8_t *)src + o->offset; - uint8_t **field_dst8 = (uint8_t **)field_dst; - uint8_t **field_src8 = (uint8_t **)field_src; - if (o->type == AV_OPT_TYPE_STRING) { - if (*field_dst8 != *field_src8) - av_freep(field_dst8); - *field_dst8 = av_strdup(*field_src8); - if (*field_src8 && !*field_dst8) - ret = AVERROR(ENOMEM); - } else if (o->type == AV_OPT_TYPE_BINARY) { - int len = *(int *)(field_src8 + 1); - if (*field_dst8 != *field_src8) - av_freep(field_dst8); - *field_dst8 = av_memdup(*field_src8, len); - if (len && !*field_dst8) { - ret = AVERROR(ENOMEM); - len = 0; - } - *(int *)(field_dst8 + 1) = len; - } else if (o->type == AV_OPT_TYPE_CONST) { - // do nothing - } else if (o->type == AV_OPT_TYPE_DICT) { - AVDictionary **sdict = (AVDictionary **) field_src; - AVDictionary **ddict = (AVDictionary **) field_dst; - int ret2; - if (*sdict != *ddict) - av_dict_free(ddict); - *ddict = NULL; - ret2 = av_dict_copy(ddict, *sdict, 0); - if (ret2 < 0) - ret = ret2; - } else if (o->type == AV_OPT_TYPE_CHLAYOUT) { - if (field_dst != field_src) - ret = av_channel_layout_copy(field_dst, field_src); - } else if (opt_is_pod(o->type)) { - size_t size = opt_elem_size[o->type]; - memcpy(field_dst, field_src, size); - } else { - av_log(dst, AV_LOG_ERROR, "Unhandled option type: %d\n", - o->type); - ret = AVERROR(EINVAL); - } + int err = opt_copy_elem(dst, o->type, field_dst, field_src); + if (err < 0) + ret = err; } return ret; }