mirror of
				https://gitlab.freedesktop.org/pipewire/pipewire.git
				synced 2025-11-03 09:01:54 -05:00 
			
		
		
		
	SPDX tags make the licensing information easy to understand and clear, and they are machine parseable. See https://spdx.dev for more information.
		
			
				
	
	
		
			145 lines
		
	
	
	
		
			3.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			145 lines
		
	
	
	
		
			3.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* Simple Plugin API */
 | 
						|
/* SPDX-FileCopyrightText: Copyright © 2021 Wim Taymans */
 | 
						|
/* SPDX-License-Identifier: MIT */
 | 
						|
 | 
						|
#include <string.h>
 | 
						|
#include <stdio.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <unistd.h>
 | 
						|
#include <sys/types.h>
 | 
						|
#include <sys/stat.h>
 | 
						|
#include <sys/mman.h>
 | 
						|
#include <fcntl.h>
 | 
						|
#include <errno.h>
 | 
						|
 | 
						|
#include <spa/utils/result.h>
 | 
						|
#include <spa/utils/json.h>
 | 
						|
 | 
						|
static void encode_string(FILE *f, const char *val, int len)
 | 
						|
{
 | 
						|
	int i;
 | 
						|
	fprintf(f, "\"");
 | 
						|
	for (i = 0; i < len; i++) {
 | 
						|
		char v = val[i];
 | 
						|
		switch (v) {
 | 
						|
		case '\n':
 | 
						|
			fprintf(f, "\\n");
 | 
						|
			break;
 | 
						|
		case '\r':
 | 
						|
			fprintf(f, "\\r");
 | 
						|
			break;
 | 
						|
		case '\b':
 | 
						|
			fprintf(f, "\\b");
 | 
						|
			break;
 | 
						|
		case '\t':
 | 
						|
			fprintf(f, "\\t");
 | 
						|
			break;
 | 
						|
		case '\f':
 | 
						|
			fprintf(f, "\\f");
 | 
						|
			break;
 | 
						|
		case '\\': case '"':
 | 
						|
			fprintf(f, "\\%c", v);
 | 
						|
			break;
 | 
						|
		default:
 | 
						|
			if (v > 0 && v < 0x20)
 | 
						|
				fprintf(f, "\\u%04x", v);
 | 
						|
			else
 | 
						|
				fprintf(f, "%c", v);
 | 
						|
			break;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	fprintf(f, "\"");
 | 
						|
}
 | 
						|
 | 
						|
static int dump(FILE *file, int indent, struct spa_json *it, const char *value, int len)
 | 
						|
{
 | 
						|
	struct spa_json sub;
 | 
						|
	int count = 0;
 | 
						|
	char key[1024];
 | 
						|
 | 
						|
	if (spa_json_is_array(value, len)) {
 | 
						|
		fprintf(file, "[");
 | 
						|
		spa_json_enter(it, &sub);
 | 
						|
		while ((len = spa_json_next(&sub, &value)) > 0) {
 | 
						|
			fprintf(file, "%s\n%*s", count++ > 0 ? "," : "",
 | 
						|
					indent+2, "");
 | 
						|
			dump(file, indent+2, &sub, value, len);
 | 
						|
		}
 | 
						|
		fprintf(file, "%s%*s]", count > 0 ? "\n" : "",
 | 
						|
				count > 0 ? indent : 0, "");
 | 
						|
	} else if (spa_json_is_object(value, len)) {
 | 
						|
		fprintf(file, "{");
 | 
						|
		spa_json_enter(it, &sub);
 | 
						|
		while (spa_json_get_string(&sub, key, sizeof(key)) > 0) {
 | 
						|
			fprintf(file, "%s\n%*s",
 | 
						|
					count++ > 0 ? "," : "",
 | 
						|
					indent+2, "");
 | 
						|
			encode_string(file, key, strlen(key));
 | 
						|
			fprintf(file, ": ");
 | 
						|
			if ((len = spa_json_next(&sub, &value)) <= 0)
 | 
						|
				break;
 | 
						|
			dump(file, indent+2, &sub, value, len);
 | 
						|
		}
 | 
						|
		fprintf(file, "%s%*s}", count > 0 ? "\n" : "",
 | 
						|
				count > 0 ? indent : 0, "");
 | 
						|
	} else if (spa_json_is_string(value, len) ||
 | 
						|
	    spa_json_is_null(value, len) ||
 | 
						|
	    spa_json_is_bool(value, len) ||
 | 
						|
	    spa_json_is_int(value, len) ||
 | 
						|
	    spa_json_is_float(value, len)) {
 | 
						|
		fprintf(file, "%.*s", len, value);
 | 
						|
	} else {
 | 
						|
		encode_string(file, value, len);
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
int main(int argc, char *argv[])
 | 
						|
{
 | 
						|
	int fd, len, res, exit_code = EXIT_FAILURE;
 | 
						|
	void *data;
 | 
						|
	struct stat sbuf;
 | 
						|
	struct spa_json it;
 | 
						|
	const char *value;
 | 
						|
 | 
						|
	if (argc < 2) {
 | 
						|
		fprintf(stderr, "usage: %s <spa-json-file>\n", argv[0]);
 | 
						|
		goto error;
 | 
						|
	}
 | 
						|
	if ((fd = open(argv[1],  O_CLOEXEC | O_RDONLY)) < 0)  {
 | 
						|
                fprintf(stderr, "error opening file '%s': %m\n", argv[1]);
 | 
						|
		goto error;
 | 
						|
        }
 | 
						|
        if (fstat(fd, &sbuf) < 0) {
 | 
						|
                fprintf(stderr, "error statting file '%s': %m\n", argv[1]);
 | 
						|
                goto error_close;
 | 
						|
	}
 | 
						|
        if ((data = mmap(NULL, sbuf.st_size, PROT_READ, MAP_PRIVATE, fd, 0)) == MAP_FAILED) {
 | 
						|
                fprintf(stderr, "error mmapping file '%s': %m\n", argv[1]);
 | 
						|
                goto error_close;
 | 
						|
	}
 | 
						|
 | 
						|
	spa_json_init(&it, data, sbuf.st_size);
 | 
						|
	if ((len = spa_json_next(&it, &value)) <= 0) {
 | 
						|
                fprintf(stderr, "not a valid file '%s': %s\n", argv[1], spa_strerror(len));
 | 
						|
		goto error_unmap;
 | 
						|
	}
 | 
						|
	if (!spa_json_is_container(value, len)) {
 | 
						|
		spa_json_init(&it, data, sbuf.st_size);
 | 
						|
		value = "{";
 | 
						|
		len = 1;
 | 
						|
	}
 | 
						|
	if ((res = dump(stdout, 0, &it, value, len)) < 0) {
 | 
						|
                fprintf(stderr, "error parsing file '%s': %s\n", argv[1], spa_strerror(res));
 | 
						|
		goto error_unmap;
 | 
						|
	}
 | 
						|
	fprintf(stdout, "\n");
 | 
						|
	exit_code = EXIT_SUCCESS;
 | 
						|
 | 
						|
error_unmap:
 | 
						|
        munmap(data, sbuf.st_size);
 | 
						|
error_close:
 | 
						|
        close(fd);
 | 
						|
error:
 | 
						|
	return exit_code;
 | 
						|
}
 |