xref: /freebsd/lib/libc/gen/sched_setaffinity.c (revision cbc32e4c5e8427e0f1032d9d3aa5863dd1538c11)
143736b71SKonstantin Belousov /*-
243736b71SKonstantin Belousov  * Copyright (c) 2021 The FreeBSD Foundation
343736b71SKonstantin Belousov  *
443736b71SKonstantin Belousov  * This software were developed by Konstantin Belousov <kib@FreeBSD.org>
543736b71SKonstantin Belousov  * under sponsorship from the FreeBSD Foundation.
643736b71SKonstantin Belousov  *
743736b71SKonstantin Belousov  * Redistribution and use in source and binary forms, with or without
843736b71SKonstantin Belousov  * modification, are permitted provided that the following conditions
943736b71SKonstantin Belousov  * are met:
1043736b71SKonstantin Belousov  * 1. Redistributions of source code must retain the above copyright
1143736b71SKonstantin Belousov  *    notice, this list of conditions and the following disclaimer.
1243736b71SKonstantin Belousov  * 2. Redistributions in binary form must reproduce the above copyright
1343736b71SKonstantin Belousov  *    notice, this list of conditions and the following disclaimer in the
1443736b71SKonstantin Belousov  *    documentation and/or other materials provided with the distribution.
1543736b71SKonstantin Belousov  *
1643736b71SKonstantin Belousov  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
1743736b71SKonstantin Belousov  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
1843736b71SKonstantin Belousov  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
1943736b71SKonstantin Belousov  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
2043736b71SKonstantin Belousov  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
2143736b71SKonstantin Belousov  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2243736b71SKonstantin Belousov  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
2343736b71SKonstantin Belousov  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
2443736b71SKonstantin Belousov  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
2543736b71SKonstantin Belousov  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
2643736b71SKonstantin Belousov  * SUCH DAMAGE.
2743736b71SKonstantin Belousov  */
2843736b71SKonstantin Belousov 
29*cbc32e4cSDmitry Chagin #define	_WANT_P_OSREL
30f35093f8SDmitry Chagin #include <sys/param.h>
31f35093f8SDmitry Chagin #include <sys/sysctl.h>
32d9cacbf4SKonstantin Belousov #include <errno.h>
3343736b71SKonstantin Belousov #include <sched.h>
34d9cacbf4SKonstantin Belousov #include <string.h>
3543736b71SKonstantin Belousov 
36*cbc32e4cSDmitry Chagin #include "libc_private.h"
37*cbc32e4cSDmitry Chagin 
3843736b71SKonstantin Belousov int
sched_setaffinity(pid_t pid,size_t cpusetsz,const cpuset_t * cpuset)3943736b71SKonstantin Belousov sched_setaffinity(pid_t pid, size_t cpusetsz, const cpuset_t *cpuset)
4043736b71SKonstantin Belousov {
41f35093f8SDmitry Chagin 	static int mp_maxid;
42*cbc32e4cSDmitry Chagin 	cpuwhich_t which;
43d9cacbf4SKonstantin Belousov 	cpuset_t c;
44f35093f8SDmitry Chagin 	int error, lbs, cpu;
45f35093f8SDmitry Chagin 	size_t len, sz;
46d9cacbf4SKonstantin Belousov 
47*cbc32e4cSDmitry Chagin 	if (__getosreldate() < P_OSREL_TIDPID) {
48*cbc32e4cSDmitry Chagin 		if (pid == 0 || pid > _PID_MAX)
49*cbc32e4cSDmitry Chagin 			which = CPU_WHICH_TID;
50*cbc32e4cSDmitry Chagin 		else
51*cbc32e4cSDmitry Chagin 			which = CPU_WHICH_PID;
52*cbc32e4cSDmitry Chagin 	} else
53*cbc32e4cSDmitry Chagin 		which = CPU_WHICH_TIDPID;
54*cbc32e4cSDmitry Chagin 
55f35093f8SDmitry Chagin 	sz = cpusetsz > sizeof(cpuset_t) ? sizeof(cpuset_t) : cpusetsz;
56d9cacbf4SKonstantin Belousov 	memset(&c, 0, sizeof(c));
57f35093f8SDmitry Chagin 	memcpy(&c, cpuset, sz);
58f35093f8SDmitry Chagin 
59f35093f8SDmitry Chagin 	/* Linux ignores high bits */
60f35093f8SDmitry Chagin 	if (mp_maxid == 0) {
61f35093f8SDmitry Chagin 		len = sizeof(mp_maxid);
62f35093f8SDmitry Chagin 		error = sysctlbyname("kern.smp.maxid", &mp_maxid, &len,
63f35093f8SDmitry Chagin 		    NULL, 0);
64f35093f8SDmitry Chagin 		if (error == -1)
65f35093f8SDmitry Chagin 			return (error);
66f35093f8SDmitry Chagin 	}
67f35093f8SDmitry Chagin 	lbs = CPU_FLS(&c) - 1;
68f35093f8SDmitry Chagin 	if (lbs > mp_maxid) {
69f35093f8SDmitry Chagin 		CPU_FOREACH_ISSET(cpu, &c)
70f35093f8SDmitry Chagin 			if (cpu > mp_maxid)
71f35093f8SDmitry Chagin 				CPU_CLR(cpu, &c);
72d9cacbf4SKonstantin Belousov 	}
73*cbc32e4cSDmitry Chagin 	error = cpuset_setaffinity(CPU_LEVEL_WHICH, which,
7467fc9502SKonstantin Belousov 	    pid == 0 ? -1 : pid, sizeof(cpuset_t), &c);
7567fc9502SKonstantin Belousov 	if (error == -1 && errno == EDEADLK)
7667fc9502SKonstantin Belousov 		errno = EINVAL;
7767fc9502SKonstantin Belousov 
7867fc9502SKonstantin Belousov 	return (error);
7943736b71SKonstantin Belousov }
80