blob: ebc182b034d75a86240f2edd42717ab29b5b1f51 [file] [log] [blame]
/*
* 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