aboutsummaryrefslogtreecommitdiff
path: root/include/qapi/clone-visitor.h
blob: ebc182b034d75a86240f2edd42717ab29b5b1f51 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
/*
 * Clone Visitor
 *
 * Copyright (C) 2016 Red Hat, Inc.
 *
 * This work is licensed under the terms of the GNU GPL, version 2 or later.
 * See the COPYING file in the top-level directory.
 *
 */

#ifndef QAPI_CLONE_VISITOR_H
#define QAPI_CLONE_VISITOR_H

#include "qapi/error.h"
#include "qapi/visitor.h"

/*
 * The clone visitor is for direct use only by the QAPI_CLONE() macro;
 * it requires that the root visit occur on an object, list, or
 * alternate, and is not usable directly on built-in QAPI types.
 */
typedef struct QapiCloneVisitor QapiCloneVisitor;

Visitor *qapi_clone_visitor_new(void);
Visitor *qapi_clone_members_visitor_new(void);

/*
 * Deep-clone QAPI object @src of the given @type, and return the result.
 *
 * Not usable on QAPI scalars (integers, strings, enums), nor on a
 * QAPI object that references the 'any' type.  Safe when @src is NULL.
 */
#define QAPI_CLONE(type, src)                                   \
    ({                                                          \
        Visitor *v_;                                            \
        type *dst_ = (type *) (src); /* Cast away const */      \
                                                                \
        if (dst_) {                                             \
            v_ = qapi_clone_visitor_new();                      \
            visit_type_ ## type(v_, NULL, &dst_, &error_abort); \
            visit_free(v_);                                     \
        }                                                       \
        dst_;                                                   \
    })

/*
 * Copy deep clones of @type members from @src to @dst.
 *
 * Not usable on QAPI scalars (integers, strings, enums), nor on a
 * QAPI object that references the 'any' type.
 */
#define QAPI_CLONE_MEMBERS(type, dst, src)                                \
    ({                                                                    \
        Visitor *v_;                                                      \
                                                                          \
        v_ = qapi_clone_members_visitor_new();                            \
        *(type *)(dst) = *(src);                                          \
        visit_type_ ## type ## _members(v_, (type *)(dst), &error_abort); \
        visit_free(v_);                                                   \
    })

#endif