/*
 * libfdt - Flat Device Tree manipulation
 *	Testcase for fdt_check_header
 * Copyright (C) 2018 David Gibson
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public License
 * as published by the Free Software Foundation; either version 2.1 of
 * the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful, but
 * WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
 */

#include <stdio.h>

#include <libfdt.h>

#include "tests.h"

static void *dtdup(void *dt)
{
	size_t bufsize = fdt_totalsize(dt);
	void *buf = xmalloc(bufsize);
	fdt_move(dt, buf, bufsize);
	return buf;
}

#define CHECK_MANGLE(exerr, code)					\
	do {								\
		void *fdt = dtdup(template);				\
		{ code }						\
		err = fdt_check_header(fdt);				\
		verbose_printf("\"%s\" => %s\n", #code, fdt_strerror(err)); \
		if (err != (exerr))					\
			FAIL("fdt_check_header() didn't catch mangle %s", \
			     #code);					\
		free(fdt);						\
	} while (0)

int main(int argc, char *argv[])
{
	void *template;
	int err;

	test_init(argc, argv);
	template = load_blob(argv[1]);

	/* Check that the base dt is valid before mangling it */
	err = fdt_check_header(template);
	if (err != 0)
		FAIL("Base tree fails: %s", fdt_strerror(err));

	/* Check a no-op mangle doesn't break things */
	CHECK_MANGLE(0, ; );

	/* Mess up the magic number */
	CHECK_MANGLE(-FDT_ERR_BADMAGIC,
		fdt_set_magic(fdt, fdt_magic(fdt) ^ 0x1);
	);
	CHECK_MANGLE(-FDT_ERR_BADMAGIC,
		fdt_set_magic(fdt, fdt_magic(fdt) ^ 0x80000000);
	);

	/* Mess up the version */
	CHECK_MANGLE(-FDT_ERR_BADVERSION,
		fdt_set_version(fdt, FDT_FIRST_SUPPORTED_VERSION - 1);
		fdt_set_last_comp_version(fdt, FDT_FIRST_SUPPORTED_VERSION - 1);
	);
	CHECK_MANGLE(-FDT_ERR_BADVERSION,
		fdt_set_version(fdt, FDT_LAST_SUPPORTED_VERSION + 1);
		fdt_set_last_comp_version(fdt, FDT_LAST_SUPPORTED_VERSION + 1);
	);
	CHECK_MANGLE(-FDT_ERR_BADVERSION,
		fdt_set_version(fdt, FDT_FIRST_SUPPORTED_VERSION);
		fdt_set_last_comp_version(fdt, FDT_LAST_SUPPORTED_VERSION);
	);

	/* Out of bounds sizes */
	CHECK_MANGLE(-FDT_ERR_TRUNCATED,
		fdt_set_totalsize(fdt, FDT_V1_SIZE - 1);
	);
	CHECK_MANGLE(-FDT_ERR_TRUNCATED,
		     fdt_set_totalsize(fdt, (uint32_t)INT_MAX + 1);
	);

	/* Truncate within various blocks */
	CHECK_MANGLE(-FDT_ERR_TRUNCATED,
		fdt_set_totalsize(fdt, fdt_off_dt_struct(fdt) - 1);
	);
	CHECK_MANGLE(-FDT_ERR_TRUNCATED,
		fdt_set_totalsize(fdt, fdt_off_dt_strings(fdt) - 1);
	);
	CHECK_MANGLE(-FDT_ERR_TRUNCATED,
		fdt_set_totalsize(fdt, fdt_off_mem_rsvmap(fdt) - 1);
	);
	CHECK_MANGLE(-FDT_ERR_TRUNCATED,
		fdt_set_totalsize(fdt, fdt_off_dt_struct(fdt) + 1);
	);
	CHECK_MANGLE(-FDT_ERR_TRUNCATED,
		fdt_set_totalsize(fdt, fdt_off_dt_strings(fdt) + 1);
	);
	CHECK_MANGLE(-FDT_ERR_TRUNCATED,
		fdt_set_totalsize(fdt, fdt_off_mem_rsvmap(fdt) + 1);
	);

	/* Negative block sizes */
	CHECK_MANGLE(-FDT_ERR_TRUNCATED,
		fdt_set_size_dt_struct(fdt, (uint32_t)-1);
	);
	CHECK_MANGLE(-FDT_ERR_TRUNCATED,
		fdt_set_size_dt_strings(fdt, (uint32_t)-1);
	);
	CHECK_MANGLE(-FDT_ERR_TRUNCATED,
		     fdt_set_size_dt_struct(fdt, (uint32_t)INT_MIN);
	);
	CHECK_MANGLE(-FDT_ERR_TRUNCATED,
		fdt_set_size_dt_strings(fdt, (uint32_t)INT_MIN);
	);

	PASS();
}