[FFmpeg-cvslog] lavu/opt: factor per-type dispatch out of av_opt_copy()
Anton Khirnov
git at videolan.org
Fri Mar 8 09:06:12 EET 2024
ffmpeg | branch: master | Anton Khirnov <anton at khirnov.net> | Mon Mar 4 10:54:58 2024 +0100| [c259f8031561662da32f8027b8503c218679af3b] | committer: Anton Khirnov
lavu/opt: factor per-type dispatch out of av_opt_copy()
Will be useful in following commits.
> http://git.videolan.org/gitweb.cgi/ffmpeg.git/?a=commit;h=c259f8031561662da32f8027b8503c218679af3b
---
libavutil/opt.c | 91 +++++++++++++++++++++++++++++++--------------------------
1 file changed, 49 insertions(+), 42 deletions(-)
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;
}
More information about the ffmpeg-cvslog
mailing list