162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * Copyright (C) 2015-2019 ARM Limited.
462306a36Sopenharmony_ci * Original author: Dave Martin <Dave.Martin@arm.com>
562306a36Sopenharmony_ci */
662306a36Sopenharmony_ci#define _GNU_SOURCE
762306a36Sopenharmony_ci#include <assert.h>
862306a36Sopenharmony_ci#include <errno.h>
962306a36Sopenharmony_ci#include <limits.h>
1062306a36Sopenharmony_ci#include <stddef.h>
1162306a36Sopenharmony_ci#include <stdio.h>
1262306a36Sopenharmony_ci#include <stdlib.h>
1362306a36Sopenharmony_ci#include <string.h>
1462306a36Sopenharmony_ci#include <getopt.h>
1562306a36Sopenharmony_ci#include <unistd.h>
1662306a36Sopenharmony_ci#include <sys/auxv.h>
1762306a36Sopenharmony_ci#include <sys/prctl.h>
1862306a36Sopenharmony_ci#include <asm/hwcap.h>
1962306a36Sopenharmony_ci#include <asm/sigcontext.h>
2062306a36Sopenharmony_ci
2162306a36Sopenharmony_cistatic int inherit = 0;
2262306a36Sopenharmony_cistatic int no_inherit = 0;
2362306a36Sopenharmony_cistatic int force = 0;
2462306a36Sopenharmony_cistatic unsigned long vl;
2562306a36Sopenharmony_cistatic int set_ctl = PR_SVE_SET_VL;
2662306a36Sopenharmony_cistatic int get_ctl = PR_SVE_GET_VL;
2762306a36Sopenharmony_ci
2862306a36Sopenharmony_cistatic const struct option options[] = {
2962306a36Sopenharmony_ci	{ "force",	no_argument, NULL, 'f' },
3062306a36Sopenharmony_ci	{ "inherit",	no_argument, NULL, 'i' },
3162306a36Sopenharmony_ci	{ "max",	no_argument, NULL, 'M' },
3262306a36Sopenharmony_ci	{ "no-inherit",	no_argument, &no_inherit, 1 },
3362306a36Sopenharmony_ci	{ "sme",	no_argument, NULL, 's' },
3462306a36Sopenharmony_ci	{ "help",	no_argument, NULL, '?' },
3562306a36Sopenharmony_ci	{}
3662306a36Sopenharmony_ci};
3762306a36Sopenharmony_ci
3862306a36Sopenharmony_cistatic char const *program_name;
3962306a36Sopenharmony_ci
4062306a36Sopenharmony_cistatic int parse_options(int argc, char **argv)
4162306a36Sopenharmony_ci{
4262306a36Sopenharmony_ci	int c;
4362306a36Sopenharmony_ci	char *rest;
4462306a36Sopenharmony_ci
4562306a36Sopenharmony_ci	program_name = strrchr(argv[0], '/');
4662306a36Sopenharmony_ci	if (program_name)
4762306a36Sopenharmony_ci		++program_name;
4862306a36Sopenharmony_ci	else
4962306a36Sopenharmony_ci		program_name = argv[0];
5062306a36Sopenharmony_ci
5162306a36Sopenharmony_ci	while ((c = getopt_long(argc, argv, "Mfhi", options, NULL)) != -1)
5262306a36Sopenharmony_ci		switch (c) {
5362306a36Sopenharmony_ci		case 'M':	vl = SVE_VL_MAX; break;
5462306a36Sopenharmony_ci		case 'f':	force = 1; break;
5562306a36Sopenharmony_ci		case 'i':	inherit = 1; break;
5662306a36Sopenharmony_ci		case 's':	set_ctl = PR_SME_SET_VL;
5762306a36Sopenharmony_ci				get_ctl = PR_SME_GET_VL;
5862306a36Sopenharmony_ci				break;
5962306a36Sopenharmony_ci		case 0:		break;
6062306a36Sopenharmony_ci		default:	goto error;
6162306a36Sopenharmony_ci		}
6262306a36Sopenharmony_ci
6362306a36Sopenharmony_ci	if (inherit && no_inherit)
6462306a36Sopenharmony_ci		goto error;
6562306a36Sopenharmony_ci
6662306a36Sopenharmony_ci	if (!vl) {
6762306a36Sopenharmony_ci		/* vector length */
6862306a36Sopenharmony_ci		if (optind >= argc)
6962306a36Sopenharmony_ci			goto error;
7062306a36Sopenharmony_ci
7162306a36Sopenharmony_ci		errno = 0;
7262306a36Sopenharmony_ci		vl = strtoul(argv[optind], &rest, 0);
7362306a36Sopenharmony_ci		if (*rest) {
7462306a36Sopenharmony_ci			vl = ULONG_MAX;
7562306a36Sopenharmony_ci			errno = EINVAL;
7662306a36Sopenharmony_ci		}
7762306a36Sopenharmony_ci		if (vl == ULONG_MAX && errno) {
7862306a36Sopenharmony_ci			fprintf(stderr, "%s: %s: %s\n",
7962306a36Sopenharmony_ci				program_name, argv[optind], strerror(errno));
8062306a36Sopenharmony_ci			goto error;
8162306a36Sopenharmony_ci		}
8262306a36Sopenharmony_ci
8362306a36Sopenharmony_ci		++optind;
8462306a36Sopenharmony_ci	}
8562306a36Sopenharmony_ci
8662306a36Sopenharmony_ci	/* command */
8762306a36Sopenharmony_ci	if (optind >= argc)
8862306a36Sopenharmony_ci		goto error;
8962306a36Sopenharmony_ci
9062306a36Sopenharmony_ci	return 0;
9162306a36Sopenharmony_ci
9262306a36Sopenharmony_cierror:
9362306a36Sopenharmony_ci	fprintf(stderr,
9462306a36Sopenharmony_ci		"Usage: %s [-f | --force] "
9562306a36Sopenharmony_ci		"[-i | --inherit | --no-inherit] "
9662306a36Sopenharmony_ci		"{-M | --max | <vector length>} "
9762306a36Sopenharmony_ci		"<command> [<arguments> ...]\n",
9862306a36Sopenharmony_ci		program_name);
9962306a36Sopenharmony_ci	return -1;
10062306a36Sopenharmony_ci}
10162306a36Sopenharmony_ci
10262306a36Sopenharmony_ciint main(int argc, char **argv)
10362306a36Sopenharmony_ci{
10462306a36Sopenharmony_ci	int ret = 126;	/* same as sh(1) command-not-executable error */
10562306a36Sopenharmony_ci	long flags;
10662306a36Sopenharmony_ci	char *path;
10762306a36Sopenharmony_ci	int t, e;
10862306a36Sopenharmony_ci
10962306a36Sopenharmony_ci	if (parse_options(argc, argv))
11062306a36Sopenharmony_ci		return 2;	/* same as sh(1) builtin incorrect-usage */
11162306a36Sopenharmony_ci
11262306a36Sopenharmony_ci	if (vl & ~(vl & PR_SVE_VL_LEN_MASK)) {
11362306a36Sopenharmony_ci		fprintf(stderr, "%s: Invalid vector length %lu\n",
11462306a36Sopenharmony_ci			program_name, vl);
11562306a36Sopenharmony_ci		return 2;	/* same as sh(1) builtin incorrect-usage */
11662306a36Sopenharmony_ci	}
11762306a36Sopenharmony_ci
11862306a36Sopenharmony_ci	if (!(getauxval(AT_HWCAP) & HWCAP_SVE)) {
11962306a36Sopenharmony_ci		fprintf(stderr, "%s: Scalable Vector Extension not present\n",
12062306a36Sopenharmony_ci			program_name);
12162306a36Sopenharmony_ci
12262306a36Sopenharmony_ci		if (!force)
12362306a36Sopenharmony_ci			goto error;
12462306a36Sopenharmony_ci
12562306a36Sopenharmony_ci		fputs("Going ahead anyway (--force):  "
12662306a36Sopenharmony_ci		      "This is a debug option.  Don't rely on it.\n",
12762306a36Sopenharmony_ci		      stderr);
12862306a36Sopenharmony_ci	}
12962306a36Sopenharmony_ci
13062306a36Sopenharmony_ci	flags = PR_SVE_SET_VL_ONEXEC;
13162306a36Sopenharmony_ci	if (inherit)
13262306a36Sopenharmony_ci		flags |= PR_SVE_VL_INHERIT;
13362306a36Sopenharmony_ci
13462306a36Sopenharmony_ci	t = prctl(set_ctl, vl | flags);
13562306a36Sopenharmony_ci	if (t < 0) {
13662306a36Sopenharmony_ci		fprintf(stderr, "%s: PR_SVE_SET_VL: %s\n",
13762306a36Sopenharmony_ci			program_name, strerror(errno));
13862306a36Sopenharmony_ci		goto error;
13962306a36Sopenharmony_ci	}
14062306a36Sopenharmony_ci
14162306a36Sopenharmony_ci	t = prctl(get_ctl);
14262306a36Sopenharmony_ci	if (t == -1) {
14362306a36Sopenharmony_ci		fprintf(stderr, "%s: PR_SVE_GET_VL: %s\n",
14462306a36Sopenharmony_ci			program_name, strerror(errno));
14562306a36Sopenharmony_ci		goto error;
14662306a36Sopenharmony_ci	}
14762306a36Sopenharmony_ci	flags = PR_SVE_VL_LEN_MASK;
14862306a36Sopenharmony_ci	flags = t & ~flags;
14962306a36Sopenharmony_ci
15062306a36Sopenharmony_ci	assert(optind < argc);
15162306a36Sopenharmony_ci	path = argv[optind];
15262306a36Sopenharmony_ci
15362306a36Sopenharmony_ci	execvp(path, &argv[optind]);
15462306a36Sopenharmony_ci	e = errno;
15562306a36Sopenharmony_ci	if (errno == ENOENT)
15662306a36Sopenharmony_ci		ret = 127;	/* same as sh(1) not-found error */
15762306a36Sopenharmony_ci	fprintf(stderr, "%s: %s: %s\n", program_name, path, strerror(e));
15862306a36Sopenharmony_ci
15962306a36Sopenharmony_cierror:
16062306a36Sopenharmony_ci	return ret;		/* same as sh(1) not-executable error */
16162306a36Sopenharmony_ci}
162