| // SPDX-License-Identifier: GPL-2.0-or-later | 
 | /* | 
 |  * (C) Copyright Linaro, Ltd. 2018 | 
 |  * (C) Copyright Arm Holdings.  2017 | 
 |  * (C) Copyright David Gibson <dwg@au1.ibm.com>, IBM Corporation.  2005. | 
 |  */ | 
 |  | 
 | #include <stdlib.h> | 
 | #include <yaml.h> | 
 | #include "dtc.h" | 
 | #include "srcpos.h" | 
 |  | 
 | char *yaml_error_name[] = { | 
 | 	[YAML_NO_ERROR] = "no error", | 
 | 	[YAML_MEMORY_ERROR] = "memory error", | 
 | 	[YAML_READER_ERROR] = "reader error", | 
 | 	[YAML_SCANNER_ERROR] = "scanner error", | 
 | 	[YAML_PARSER_ERROR] = "parser error", | 
 | 	[YAML_COMPOSER_ERROR] = "composer error", | 
 | 	[YAML_WRITER_ERROR] = "writer error", | 
 | 	[YAML_EMITTER_ERROR] = "emitter error", | 
 | }; | 
 |  | 
 | #define yaml_emitter_emit_or_die(emitter, event) (			\ | 
 | {									\ | 
 | 	if (!yaml_emitter_emit(emitter, event))				\ | 
 | 		die("yaml '%s': %s in %s, line %i\n",			\ | 
 | 		    yaml_error_name[(emitter)->error], 			\ | 
 | 		    (emitter)->problem, __func__, __LINE__);		\ | 
 | }) | 
 |  | 
 | static void yaml_propval_int(yaml_emitter_t *emitter, struct marker *markers, char *data, int len, int width) | 
 | { | 
 | 	yaml_event_t event; | 
 | 	void *tag; | 
 | 	int off, start_offset = markers->offset; | 
 |  | 
 | 	switch(width) { | 
 | 		case 1: tag = "!u8"; break; | 
 | 		case 2: tag = "!u16"; break; | 
 | 		case 4: tag = "!u32"; break; | 
 | 		case 8: tag = "!u64"; break; | 
 | 		default: | 
 | 			die("Invalid width %i", width); | 
 | 	} | 
 | 	assert(len % width == 0); | 
 |  | 
 | 	yaml_sequence_start_event_initialize(&event, NULL, | 
 | 		(yaml_char_t *)tag, width == 4, YAML_FLOW_SEQUENCE_STYLE); | 
 | 	yaml_emitter_emit_or_die(emitter, &event); | 
 |  | 
 | 	for (off = 0; off < len; off += width) { | 
 | 		char buf[32]; | 
 | 		struct marker *m; | 
 | 		bool is_phandle = false; | 
 |  | 
 | 		switch(width) { | 
 | 		case 1: | 
 | 			sprintf(buf, "0x%"PRIx8, *(uint8_t*)(data + off)); | 
 | 			break; | 
 | 		case 2: | 
 | 			sprintf(buf, "0x%"PRIx16, fdt16_to_cpu(*(fdt16_t*)(data + off))); | 
 | 			break; | 
 | 		case 4: | 
 | 			sprintf(buf, "0x%"PRIx32, fdt32_to_cpu(*(fdt32_t*)(data + off))); | 
 | 			m = markers; | 
 | 			is_phandle = false; | 
 | 			for_each_marker_of_type(m, REF_PHANDLE) { | 
 | 				if (m->offset == (start_offset + off)) { | 
 | 					is_phandle = true; | 
 | 					break; | 
 | 				} | 
 | 			} | 
 | 			break; | 
 | 		case 8: | 
 | 			sprintf(buf, "0x%"PRIx64, fdt64_to_cpu(*(fdt64_t*)(data + off))); | 
 | 			break; | 
 | 		} | 
 |  | 
 | 		if (is_phandle) | 
 | 			yaml_scalar_event_initialize(&event, NULL, | 
 | 				(yaml_char_t*)"!phandle", (yaml_char_t *)buf, | 
 | 				strlen(buf), 0, 0, YAML_PLAIN_SCALAR_STYLE); | 
 | 		else | 
 | 			yaml_scalar_event_initialize(&event, NULL, | 
 | 				(yaml_char_t*)YAML_INT_TAG, (yaml_char_t *)buf, | 
 | 				strlen(buf), 1, 1, YAML_PLAIN_SCALAR_STYLE); | 
 | 		yaml_emitter_emit_or_die(emitter, &event); | 
 | 	} | 
 |  | 
 | 	yaml_sequence_end_event_initialize(&event); | 
 | 	yaml_emitter_emit_or_die(emitter, &event); | 
 | } | 
 |  | 
 | static void yaml_propval_string(yaml_emitter_t *emitter, char *str, int len) | 
 | { | 
 | 	yaml_event_t event; | 
 | 	int i; | 
 |  | 
 | 	assert(str[len-1] == '\0'); | 
 |  | 
 | 	/* Make sure the entire string is in the lower 7-bit ascii range */ | 
 | 	for (i = 0; i < len; i++) | 
 | 		assert(isascii(str[i])); | 
 |  | 
 | 	yaml_scalar_event_initialize(&event, NULL, | 
 | 		(yaml_char_t *)YAML_STR_TAG, (yaml_char_t*)str, | 
 | 		len-1, 0, 1, YAML_DOUBLE_QUOTED_SCALAR_STYLE); | 
 | 	yaml_emitter_emit_or_die(emitter, &event); | 
 | } | 
 |  | 
 | static void yaml_propval(yaml_emitter_t *emitter, struct property *prop) | 
 | { | 
 | 	yaml_event_t event; | 
 | 	int len = prop->val.len; | 
 | 	struct marker *m = prop->val.markers; | 
 |  | 
 | 	/* Emit the property name */ | 
 | 	yaml_scalar_event_initialize(&event, NULL, | 
 | 		(yaml_char_t *)YAML_STR_TAG, (yaml_char_t*)prop->name, | 
 | 		strlen(prop->name), 1, 1, YAML_PLAIN_SCALAR_STYLE); | 
 | 	yaml_emitter_emit_or_die(emitter, &event); | 
 |  | 
 | 	/* Boolean properties are easiest to deal with. Length is zero, so just emit 'true' */ | 
 | 	if (len == 0) { | 
 | 		yaml_scalar_event_initialize(&event, NULL, | 
 | 			(yaml_char_t *)YAML_BOOL_TAG, | 
 | 			(yaml_char_t*)"true", | 
 | 			strlen("true"), 1, 0, YAML_PLAIN_SCALAR_STYLE); | 
 | 		yaml_emitter_emit_or_die(emitter, &event); | 
 | 		return; | 
 | 	} | 
 |  | 
 | 	if (!m) | 
 | 		die("No markers present in property '%s' value\n", prop->name); | 
 |  | 
 | 	yaml_sequence_start_event_initialize(&event, NULL, | 
 | 		(yaml_char_t *)YAML_SEQ_TAG, 1, YAML_FLOW_SEQUENCE_STYLE); | 
 | 	yaml_emitter_emit_or_die(emitter, &event); | 
 |  | 
 | 	for_each_marker(m) { | 
 | 		int chunk_len; | 
 | 		char *data = &prop->val.val[m->offset]; | 
 |  | 
 | 		if (m->type < TYPE_UINT8) | 
 | 			continue; | 
 |  | 
 | 		chunk_len = type_marker_length(m) ? : len; | 
 | 		assert(chunk_len > 0); | 
 | 		len -= chunk_len; | 
 |  | 
 | 		switch(m->type) { | 
 | 		case TYPE_UINT16: | 
 | 			yaml_propval_int(emitter, m, data, chunk_len, 2); | 
 | 			break; | 
 | 		case TYPE_UINT32: | 
 | 			yaml_propval_int(emitter, m, data, chunk_len, 4); | 
 | 			break; | 
 | 		case TYPE_UINT64: | 
 | 			yaml_propval_int(emitter, m, data, chunk_len, 8); | 
 | 			break; | 
 | 		case TYPE_STRING: | 
 | 			yaml_propval_string(emitter, data, chunk_len); | 
 | 			break; | 
 | 		default: | 
 | 			yaml_propval_int(emitter, m, data, chunk_len, 1); | 
 | 			break; | 
 | 		} | 
 | 	} | 
 |  | 
 | 	yaml_sequence_end_event_initialize(&event); | 
 | 	yaml_emitter_emit_or_die(emitter, &event); | 
 | } | 
 |  | 
 |  | 
 | static void yaml_tree(struct node *tree, yaml_emitter_t *emitter) | 
 | { | 
 | 	struct property *prop; | 
 | 	struct node *child; | 
 | 	yaml_event_t event; | 
 |  | 
 | 	if (tree->deleted) | 
 | 		return; | 
 |  | 
 | 	yaml_mapping_start_event_initialize(&event, NULL, | 
 | 		(yaml_char_t *)YAML_MAP_TAG, 1, YAML_ANY_MAPPING_STYLE); | 
 | 	yaml_emitter_emit_or_die(emitter, &event); | 
 |  | 
 | 	for_each_property(tree, prop) | 
 | 		yaml_propval(emitter, prop); | 
 |  | 
 | 	/* Loop over all the children, emitting them into the map */ | 
 | 	for_each_child(tree, child) { | 
 | 		yaml_scalar_event_initialize(&event, NULL, | 
 | 			(yaml_char_t *)YAML_STR_TAG, (yaml_char_t*)child->name, | 
 | 			strlen(child->name), 1, 0, YAML_PLAIN_SCALAR_STYLE); | 
 | 		yaml_emitter_emit_or_die(emitter, &event); | 
 | 		yaml_tree(child, emitter); | 
 | 	} | 
 |  | 
 | 	yaml_mapping_end_event_initialize(&event); | 
 | 	yaml_emitter_emit_or_die(emitter, &event); | 
 | } | 
 |  | 
 | void dt_to_yaml(FILE *f, struct dt_info *dti) | 
 | { | 
 | 	yaml_emitter_t emitter; | 
 | 	yaml_event_t event; | 
 |  | 
 | 	yaml_emitter_initialize(&emitter); | 
 | 	yaml_emitter_set_output_file(&emitter, f); | 
 | 	yaml_stream_start_event_initialize(&event, YAML_UTF8_ENCODING); | 
 | 	yaml_emitter_emit_or_die(&emitter, &event); | 
 |  | 
 | 	yaml_document_start_event_initialize(&event, NULL, NULL, NULL, 0); | 
 | 	yaml_emitter_emit_or_die(&emitter, &event); | 
 |  | 
 | 	yaml_sequence_start_event_initialize(&event, NULL, (yaml_char_t *)YAML_SEQ_TAG, 1, YAML_ANY_SEQUENCE_STYLE); | 
 | 	yaml_emitter_emit_or_die(&emitter, &event); | 
 |  | 
 | 	yaml_tree(dti->dt, &emitter); | 
 |  | 
 | 	yaml_sequence_end_event_initialize(&event); | 
 | 	yaml_emitter_emit_or_die(&emitter, &event); | 
 |  | 
 | 	yaml_document_end_event_initialize(&event, 0); | 
 | 	yaml_emitter_emit_or_die(&emitter, &event); | 
 |  | 
 | 	yaml_stream_end_event_initialize(&event); | 
 | 	yaml_emitter_emit_or_die(&emitter, &event); | 
 |  | 
 | 	yaml_emitter_delete(&emitter); | 
 | } |