Commit 05687352 authored by Martin KaFai Lau's avatar Martin KaFai Lau Committed by Alexei Starovoitov
Browse files

bpf: Refactor and bug fix in test_func_type in test_btf.c



1) bpf_load_program_xattr() is absorbing the EBIG error
   which makes testing this case impossible.  It is replaced
   with a direct syscall(__NR_bpf, BPF_PROG_LOAD,...).
2) The test_func_type() is renamed to test_info_raw() to
   prepare for the new line_info test in the next patch.
3) The bpf_obj_get_info_by_fd() testing for func_info
   is refactored to test_get_finfo().  A new
   test_get_linfo() will be added in the next patch
   for testing line_info purpose.
4) The test->func_info_cnt is checked instead of
   a static value "2".
5) Remove unnecessary "\n" in error message.
6) Adding back info_raw_test_num to the cmd arg such
   that a specific test case can be tested, like
   all other existing tests.

7) Fix a bug in handling expected_prog_load_failure.
   A test could pass even if prog_fd != -1 while
   expected_prog_load_failure is true.
8) The min rec_size check should be < 8 instead of < 4.

Fixes: 4798c4ba ("tools/bpf: extends test_btf to test load/retrieve func_type info")
Signed-off-by: default avatarMartin KaFai Lau <kafai@fb.com>
Acked-by: default avatarYonghong Song <yhs@fb.com>
Signed-off-by: default avatarAlexei Starovoitov <ast@kernel.org>
parent ee491d8d
Loading
Loading
Loading
Loading
+125 −86
Original line number Original line Diff line number Diff line
@@ -6,6 +6,7 @@
#include <linux/err.h>
#include <linux/err.h>
#include <linux/kernel.h>
#include <linux/kernel.h>
#include <linux/filter.h>
#include <linux/filter.h>
#include <linux/unistd.h>
#include <bpf/bpf.h>
#include <bpf/bpf.h>
#include <sys/resource.h>
#include <sys/resource.h>
#include <libelf.h>
#include <libelf.h>
@@ -114,12 +115,13 @@ static struct args {
	unsigned int raw_test_num;
	unsigned int raw_test_num;
	unsigned int file_test_num;
	unsigned int file_test_num;
	unsigned int get_info_test_num;
	unsigned int get_info_test_num;
	unsigned int info_raw_test_num;
	bool raw_test;
	bool raw_test;
	bool file_test;
	bool file_test;
	bool get_info_test;
	bool get_info_test;
	bool pprint_test;
	bool pprint_test;
	bool always_log;
	bool always_log;
	bool func_type_test;
	bool info_raw_test;
} args;
} args;


static char btf_log_buf[BTF_LOG_BUF_SIZE];
static char btf_log_buf[BTF_LOG_BUF_SIZE];
@@ -3051,7 +3053,7 @@ static int test_pprint(void)
	return err;
	return err;
}
}


static struct btf_func_type_test {
static struct prog_info_raw_test {
	const char *descr;
	const char *descr;
	const char *str_sec;
	const char *str_sec;
	__u32 raw_types[MAX_NR_RAW_TYPES];
	__u32 raw_types[MAX_NR_RAW_TYPES];
@@ -3062,7 +3064,7 @@ static struct btf_func_type_test {
	__u32 func_info_rec_size;
	__u32 func_info_rec_size;
	__u32 func_info_cnt;
	__u32 func_info_cnt;
	bool expected_prog_load_failure;
	bool expected_prog_load_failure;
} func_type_test[] = {
} info_raw_tests[] = {
{
{
	.descr = "func_type (main func + one sub)",
	.descr = "func_type (main func + one sub)",
	.raw_types = {
	.raw_types = {
@@ -3198,90 +3200,44 @@ static size_t probe_prog_length(const struct bpf_insn *fp)
	return len + 1;
	return len + 1;
}
}


static int do_test_func_type(int test_num)
static int test_get_finfo(const struct prog_info_raw_test *test,
			  int prog_fd)
{
{
	const struct btf_func_type_test *test = &func_type_test[test_num];
	unsigned int raw_btf_size, info_len, rec_size;
	int i, btf_fd = -1, prog_fd = -1, err = 0;
	struct bpf_load_program_attr attr = {};
	void *raw_btf, *func_info = NULL;
	struct bpf_prog_info info = {};
	struct bpf_prog_info info = {};
	struct bpf_func_info *finfo;
	struct bpf_func_info *finfo;

	__u32 info_len, rec_size, i;
	fprintf(stderr, "%s......", test->descr);
	void *func_info = NULL;
	raw_btf = btf_raw_create(&hdr_tmpl, test->raw_types,
	int err;
				 test->str_sec, test->str_sec_size,
				 &raw_btf_size);

	if (!raw_btf)
		return -1;

	*btf_log_buf = '\0';
	btf_fd = bpf_load_btf(raw_btf, raw_btf_size,
			      btf_log_buf, BTF_LOG_BUF_SIZE,
			      args.always_log);
	free(raw_btf);

	if (CHECK(btf_fd == -1, "invalid btf_fd errno:%d", errno)) {
		err = -1;
		goto done;
	}

	if (*btf_log_buf && args.always_log)
		fprintf(stderr, "\n%s", btf_log_buf);

	attr.prog_type = test->prog_type;
	attr.insns = test->insns;
	attr.insns_cnt = probe_prog_length(attr.insns);
	attr.license = "GPL";
	attr.prog_btf_fd = btf_fd;
	attr.func_info_rec_size = test->func_info_rec_size;
	attr.func_info_cnt = test->func_info_cnt;
	attr.func_info = test->func_info;

	*btf_log_buf = '\0';
	prog_fd = bpf_load_program_xattr(&attr, btf_log_buf,
					 BTF_LOG_BUF_SIZE);
	if (test->expected_prog_load_failure && prog_fd == -1) {
		err = 0;
		goto done;
	}
	if (CHECK(prog_fd == -1, "invalid prog_id errno:%d", errno)) {
		fprintf(stderr, "%s\n", btf_log_buf);
		err = -1;
		goto done;
	}


	/* get necessary lens */
	/* get necessary lens */
	info_len = sizeof(struct bpf_prog_info);
	info_len = sizeof(struct bpf_prog_info);
	err = bpf_obj_get_info_by_fd(prog_fd, &info, &info_len);
	err = bpf_obj_get_info_by_fd(prog_fd, &info, &info_len);
	if (CHECK(err == -1, "invalid get info (1st) errno:%d", errno)) {
	if (CHECK(err == -1, "invalid get info (1st) errno:%d", errno)) {
		fprintf(stderr, "%s\n", btf_log_buf);
		fprintf(stderr, "%s\n", btf_log_buf);
		err = -1;
		return -1;
		goto done;
	}
	}
	if (CHECK(info.func_info_cnt != 2,
	if (CHECK(info.func_info_cnt != test->func_info_cnt,
		  "incorrect info.func_info_cnt (1st) %d\n",
		  "incorrect info.func_info_cnt (1st) %d",
		  info.func_info_cnt)) {
		  info.func_info_cnt)) {
		err = -1;
		return -1;
		goto done;
	}
	}

	rec_size = info.func_info_rec_size;
	rec_size = info.func_info_rec_size;
	if (CHECK(rec_size < 4,
	if (CHECK(rec_size < 8,
		  "incorrect info.func_info_rec_size (1st) %d\n", rec_size)) {
		  "incorrect info.func_info_rec_size (1st) %d", rec_size)) {
		err = -1;
		return -1;
		goto done;
	}
	}


	if (!info.func_info_cnt)
		return 0;

	func_info = malloc(info.func_info_cnt * rec_size);
	func_info = malloc(info.func_info_cnt * rec_size);
	if (CHECK(!func_info, "out of memory")) {
	if (CHECK(!func_info, "out of memory"))
		err = -1;
		return -1;
		goto done;
	}


	/* reset info to only retrieve func_info related data */
	/* reset info to only retrieve func_info related data */
	memset(&info, 0, sizeof(info));
	memset(&info, 0, sizeof(info));
	info.func_info_cnt = 2;
	info.func_info_cnt = test->func_info_cnt;
	info.func_info_rec_size = rec_size;
	info.func_info_rec_size = rec_size;
	info.func_info = ptr_to_u64(func_info);
	info.func_info = ptr_to_u64(func_info);
	err = bpf_obj_get_info_by_fd(prog_fd, &info, &info_len);
	err = bpf_obj_get_info_by_fd(prog_fd, &info, &info_len);
@@ -3290,14 +3246,14 @@ static int do_test_func_type(int test_num)
		err = -1;
		err = -1;
		goto done;
		goto done;
	}
	}
	if (CHECK(info.func_info_cnt != 2,
	if (CHECK(info.func_info_cnt != test->func_info_cnt,
		  "incorrect info.func_info_cnt (2nd) %d\n",
		  "incorrect info.func_info_cnt (2nd) %d",
		  info.func_info_cnt)) {
		  info.func_info_cnt)) {
		err = -1;
		err = -1;
		goto done;
		goto done;
	}
	}
	if (CHECK(info.func_info_rec_size != rec_size,
	if (CHECK(info.func_info_rec_size < 8,
		  "incorrect info.func_info_rec_size (2nd) %d\n",
		  "incorrect info.func_info_rec_size (2nd) %d",
		  info.func_info_rec_size)) {
		  info.func_info_rec_size)) {
		err = -1;
		err = -1;
		goto done;
		goto done;
@@ -3310,7 +3266,7 @@ static int do_test_func_type(int test_num)
	}
	}


	finfo = func_info;
	finfo = func_info;
	for (i = 0; i < 2; i++) {
	for (i = 0; i < test->func_info_cnt; i++) {
		if (CHECK(finfo->type_id != test->func_info[i][1],
		if (CHECK(finfo->type_id != test->func_info[i][1],
			  "incorrect func_type %u expected %u",
			  "incorrect func_type %u expected %u",
			  finfo->type_id, test->func_info[i][1])) {
			  finfo->type_id, test->func_info[i][1])) {
@@ -3320,7 +3276,75 @@ static int do_test_func_type(int test_num)
		finfo = (void *)finfo + rec_size;
		finfo = (void *)finfo + rec_size;
	}
	}


	err = 0;

done:
done:
	free(func_info);
	return err;
}

static int do_test_info_raw(unsigned int test_num)
{
	const struct prog_info_raw_test *test = &info_raw_tests[test_num - 1];
	int btf_fd = -1, prog_fd = -1, err = 0;
	unsigned int raw_btf_size;
	union bpf_attr attr = {};
	void *raw_btf;

	fprintf(stderr, "BTF prog info raw test[%u] (%s): ", test_num, test->descr);
	raw_btf = btf_raw_create(&hdr_tmpl, test->raw_types,
				 test->str_sec, test->str_sec_size,
				 &raw_btf_size);

	if (!raw_btf)
		return -1;

	*btf_log_buf = '\0';
	btf_fd = bpf_load_btf(raw_btf, raw_btf_size,
			      btf_log_buf, BTF_LOG_BUF_SIZE,
			      args.always_log);
	free(raw_btf);

	if (CHECK(btf_fd == -1, "invalid btf_fd errno:%d", errno)) {
		err = -1;
		goto done;
	}

	if (*btf_log_buf && args.always_log)
		fprintf(stderr, "\n%s", btf_log_buf);
	*btf_log_buf = '\0';

	attr.prog_type = test->prog_type;
	attr.insns = ptr_to_u64(test->insns);
	attr.insn_cnt = probe_prog_length(test->insns);
	attr.license = ptr_to_u64("GPL");
	attr.prog_btf_fd = btf_fd;
	attr.func_info_rec_size = test->func_info_rec_size;
	attr.func_info_cnt = test->func_info_cnt;
	attr.func_info = ptr_to_u64(test->func_info);
	attr.log_buf = ptr_to_u64(btf_log_buf);
	attr.log_size = BTF_LOG_BUF_SIZE;
	attr.log_level = 1;

	prog_fd = syscall(__NR_bpf, BPF_PROG_LOAD, &attr, sizeof(attr));
	err = ((prog_fd == -1) != test->expected_prog_load_failure);
	if (CHECK(err, "prog_fd:%d expected_prog_load_failure:%u errno:%d",
		  prog_fd, test->expected_prog_load_failure, errno)) {
		err = -1;
		goto done;
	}

	if (prog_fd == -1)
		goto done;

	err = test_get_finfo(test, prog_fd);
	if (err)
		goto done;

done:
	if (!err)
		fprintf(stderr, "OK");

	if (*btf_log_buf && (err || args.always_log))
	if (*btf_log_buf && (err || args.always_log))
		fprintf(stderr, "\n%s", btf_log_buf);
		fprintf(stderr, "\n%s", btf_log_buf);


@@ -3328,33 +3352,38 @@ done:
		close(btf_fd);
		close(btf_fd);
	if (prog_fd != -1)
	if (prog_fd != -1)
		close(prog_fd);
		close(prog_fd);
	free(func_info);

	return err;
	return err;
}
}


static int test_func_type(void)
static int test_info_raw(void)
{
{
	unsigned int i;
	unsigned int i;
	int err = 0;
	int err = 0;


	for (i = 0; i < ARRAY_SIZE(func_type_test); i++)
	if (args.info_raw_test_num)
		err |= count_result(do_test_func_type(i));
		return count_result(do_test_info_raw(args.info_raw_test_num));

	for (i = 1; i <= ARRAY_SIZE(info_raw_tests); i++)
		err |= count_result(do_test_info_raw(i));


	return err;
	return err;
}
}


static void usage(const char *cmd)
static void usage(const char *cmd)
{
{
	fprintf(stderr, "Usage: %s [-l] [[-r test_num (1 - %zu)] |"
	fprintf(stderr, "Usage: %s [-l] [[-r btf_raw_test_num (1 - %zu)] |\n"
			" [-g test_num (1 - %zu)] |"
			"\t[-g btf_get_info_test_num (1 - %zu)] |\n"
			" [-f test_num (1 - %zu)] | [-p] | [-k] ]\n",
			"\t[-f btf_file_test_num (1 - %zu)] |\n"
			"\t[-k btf_prog_info_raw_test_num (1 - %zu)] |\n"
			"\t[-p (pretty print test)]]\n",
		cmd, ARRAY_SIZE(raw_tests), ARRAY_SIZE(get_info_tests),
		cmd, ARRAY_SIZE(raw_tests), ARRAY_SIZE(get_info_tests),
		ARRAY_SIZE(file_tests));
		ARRAY_SIZE(file_tests), ARRAY_SIZE(info_raw_tests));
}
}


static int parse_args(int argc, char **argv)
static int parse_args(int argc, char **argv)
{
{
	const char *optstr = "lpkf:r:g:";
	const char *optstr = "lpk:f:r:g:";
	int opt;
	int opt;


	while ((opt = getopt(argc, argv, optstr)) != -1) {
	while ((opt = getopt(argc, argv, optstr)) != -1) {
@@ -3378,7 +3407,8 @@ static int parse_args(int argc, char **argv)
			args.pprint_test = true;
			args.pprint_test = true;
			break;
			break;
		case 'k':
		case 'k':
			args.func_type_test = true;
			args.info_raw_test_num = atoi(optarg);
			args.info_raw_test = true;
			break;
			break;
		case 'h':
		case 'h':
			usage(argv[0]);
			usage(argv[0]);
@@ -3413,6 +3443,14 @@ static int parse_args(int argc, char **argv)
		return -1;
		return -1;
	}
	}


	if (args.info_raw_test_num &&
	    (args.info_raw_test_num < 1 ||
	     args.info_raw_test_num > ARRAY_SIZE(info_raw_tests))) {
		fprintf(stderr, "BTF prog info raw test number must be [1 - %zu]\n",
			ARRAY_SIZE(info_raw_tests));
		return -1;
	}

	return 0;
	return 0;
}
}


@@ -3445,16 +3483,17 @@ int main(int argc, char **argv)
	if (args.pprint_test)
	if (args.pprint_test)
		err |= test_pprint();
		err |= test_pprint();


	if (args.func_type_test)
	if (args.info_raw_test)
		err |= test_func_type();
		err |= test_info_raw();


	if (args.raw_test || args.get_info_test || args.file_test ||
	if (args.raw_test || args.get_info_test || args.file_test ||
	    args.pprint_test || args.func_type_test)
	    args.pprint_test || args.info_raw_test)
		goto done;
		goto done;


	err |= test_raw();
	err |= test_raw();
	err |= test_get_info();
	err |= test_get_info();
	err |= test_file();
	err |= test_file();
	err |= test_info_raw();


done:
done:
	print_summary();
	print_summary();