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