| /* |
| * 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 |