| /* |
| * JSON Writer |
| * |
| * Copyright IBM, Corp. 2009 |
| * Copyright (c) 2010-2020 Red Hat Inc. |
| * |
| * Authors: |
| * Anthony Liguori <aliguori@us.ibm.com> |
| * Markus Armbruster <armbru@redhat.com> |
| * |
| * This work is licensed under the terms of the GNU LGPL, version 2.1 or later. |
| * See the COPYING.LIB file in the top-level directory. |
| * |
| */ |
| |
| #include "qemu/osdep.h" |
| #include "qapi/qmp/json-writer.h" |
| #include "qemu/unicode.h" |
| |
| struct JSONWriter { |
| bool pretty; |
| bool need_comma; |
| GString *contents; |
| GByteArray *container_is_array; |
| }; |
| |
| JSONWriter *json_writer_new(bool pretty) |
| { |
| JSONWriter *writer = g_new(JSONWriter, 1); |
| |
| writer->pretty = pretty; |
| writer->need_comma = false; |
| writer->contents = g_string_new(NULL); |
| writer->container_is_array = g_byte_array_new(); |
| return writer; |
| } |
| |
| const char *json_writer_get(JSONWriter *writer) |
| { |
| g_assert(!writer->container_is_array->len); |
| return writer->contents->str; |
| } |
| |
| GString *json_writer_get_and_free(JSONWriter *writer) |
| { |
| GString *contents = writer->contents; |
| |
| writer->contents = NULL; |
| g_byte_array_free(writer->container_is_array, true); |
| g_free(writer); |
| return contents; |
| } |
| |
| void json_writer_free(JSONWriter *writer) |
| { |
| if (writer) { |
| g_string_free(json_writer_get_and_free(writer), true); |
| } |
| } |
| |
| static void enter_container(JSONWriter *writer, bool is_array) |
| { |
| unsigned depth = writer->container_is_array->len; |
| |
| g_byte_array_set_size(writer->container_is_array, depth + 1); |
| writer->container_is_array->data[depth] = is_array; |
| writer->need_comma = false; |
| } |
| |
| static void leave_container(JSONWriter *writer, bool is_array) |
| { |
| unsigned depth = writer->container_is_array->len; |
| |
| assert(depth); |
| assert(writer->container_is_array->data[depth - 1] == is_array); |
| g_byte_array_set_size(writer->container_is_array, depth - 1); |
| writer->need_comma = true; |
| } |
| |
| static bool in_object(JSONWriter *writer) |
| { |
| unsigned depth = writer->container_is_array->len; |
| |
| return depth && !writer->container_is_array->data[depth - 1]; |
| } |
| |
| static void pretty_newline(JSONWriter *writer) |
| { |
| if (writer->pretty) { |
| g_string_append_printf(writer->contents, "\n%*s", |
| writer->container_is_array->len * 4, ""); |
| } |
| } |
| |
| static void pretty_newline_or_space(JSONWriter *writer) |
| { |
| if (writer->pretty) { |
| g_string_append_printf(writer->contents, "\n%*s", |
| writer->container_is_array->len * 4, ""); |
| } else { |
| g_string_append_c(writer->contents, ' '); |
| } |
| } |
| |
| static void quoted_str(JSONWriter *writer, const char *str) |
| { |
| const char *ptr; |
| char *end; |
| int cp; |
| |
| g_string_append_c(writer->contents, '"'); |
| |
| for (ptr = str; *ptr; ptr = end) { |
| cp = mod_utf8_codepoint(ptr, 6, &end); |
| switch (cp) { |
| case '\"': |
| g_string_append(writer->contents, "\\\""); |
| break; |
| case '\\': |
| g_string_append(writer->contents, "\\\\"); |
| break; |
| case '\b': |
| g_string_append(writer->contents, "\\b"); |
| break; |
| case '\f': |
| g_string_append(writer->contents, "\\f"); |
| break; |
| case '\n': |
| g_string_append(writer->contents, "\\n"); |
| break; |
| case '\r': |
| g_string_append(writer->contents, "\\r"); |
| break; |
| case '\t': |
| g_string_append(writer->contents, "\\t"); |
| break; |
| default: |
| if (cp < 0) { |
| cp = 0xFFFD; /* replacement character */ |
| } |
| if (cp > 0xFFFF) { |
| /* beyond BMP; need a surrogate pair */ |
| g_string_append_printf(writer->contents, "\\u%04X\\u%04X", |
| 0xD800 + ((cp - 0x10000) >> 10), |
| 0xDC00 + ((cp - 0x10000) & 0x3FF)); |
| } else if (cp < 0x20 || cp >= 0x7F) { |
| g_string_append_printf(writer->contents, "\\u%04X", cp); |
| } else { |
| g_string_append_c(writer->contents, cp); |
| } |
| } |
| }; |
| |
| g_string_append_c(writer->contents, '"'); |
| } |
| |
| static void maybe_comma_name(JSONWriter *writer, const char *name) |
| { |
| if (writer->need_comma) { |
| g_string_append_c(writer->contents, ','); |
| pretty_newline_or_space(writer); |
| } else { |
| if (writer->contents->len) { |
| pretty_newline(writer); |
| } |
| writer->need_comma = true; |
| } |
| |
| if (in_object(writer)) { |
| quoted_str(writer, name); |
| g_string_append(writer->contents, ": "); |
| } |
| } |
| |
| void json_writer_start_object(JSONWriter *writer, const char *name) |
| { |
| maybe_comma_name(writer, name); |
| g_string_append_c(writer->contents, '{'); |
| enter_container(writer, false); |
| } |
| |
| void json_writer_end_object(JSONWriter *writer) |
| { |
| leave_container(writer, false); |
| pretty_newline(writer); |
| g_string_append_c(writer->contents, '}'); |
| } |
| |
| void json_writer_start_array(JSONWriter *writer, const char *name) |
| { |
| maybe_comma_name(writer, name); |
| g_string_append_c(writer->contents, '['); |
| enter_container(writer, true); |
| } |
| |
| void json_writer_end_array(JSONWriter *writer) |
| { |
| leave_container(writer, true); |
| pretty_newline(writer); |
| g_string_append_c(writer->contents, ']'); |
| } |
| |
| void json_writer_bool(JSONWriter *writer, const char *name, bool val) |
| { |
| maybe_comma_name(writer, name); |
| g_string_append(writer->contents, val ? "true" : "false"); |
| } |
| |
| void json_writer_null(JSONWriter *writer, const char *name) |
| { |
| maybe_comma_name(writer, name); |
| g_string_append(writer->contents, "null"); |
| } |
| |
| void json_writer_int64(JSONWriter *writer, const char *name, int64_t val) |
| { |
| maybe_comma_name(writer, name); |
| g_string_append_printf(writer->contents, "%" PRId64, val); |
| } |
| |
| void json_writer_uint64(JSONWriter *writer, const char *name, uint64_t val) |
| { |
| maybe_comma_name(writer, name); |
| g_string_append_printf(writer->contents, "%" PRIu64, val); |
| } |
| |
| void json_writer_double(JSONWriter *writer, const char *name, double val) |
| { |
| maybe_comma_name(writer, name); |
| |
| /* |
| * FIXME: g_string_append_printf() is locale dependent; but JSON |
| * requires numbers to be formatted as if in the C locale. |
| * Dependence on C locale is a pervasive issue in QEMU. |
| */ |
| /* |
| * FIXME: This risks printing Inf or NaN, which are not valid |
| * JSON values. |
| */ |
| g_string_append_printf(writer->contents, "%.17g", val); |
| } |
| |
| void json_writer_str(JSONWriter *writer, const char *name, const char *str) |
| { |
| maybe_comma_name(writer, name); |
| quoted_str(writer, str); |
| } |