10d163575Sopenharmony_ci/*
20d163575Sopenharmony_ci * Copyright (c) 2013-2019 Huawei Technologies Co., Ltd. All rights reserved.
30d163575Sopenharmony_ci * Copyright (c) 2020-2021 Huawei Device Co., Ltd. All rights reserved.
40d163575Sopenharmony_ci *
50d163575Sopenharmony_ci * Redistribution and use in source and binary forms, with or without modification,
60d163575Sopenharmony_ci * are permitted provided that the following conditions are met:
70d163575Sopenharmony_ci *
80d163575Sopenharmony_ci * 1. Redistributions of source code must retain the above copyright notice, this list of
90d163575Sopenharmony_ci *    conditions and the following disclaimer.
100d163575Sopenharmony_ci *
110d163575Sopenharmony_ci * 2. Redistributions in binary form must reproduce the above copyright notice, this list
120d163575Sopenharmony_ci *    of conditions and the following disclaimer in the documentation and/or other materials
130d163575Sopenharmony_ci *    provided with the distribution.
140d163575Sopenharmony_ci *
150d163575Sopenharmony_ci * 3. Neither the name of the copyright holder nor the names of its contributors may be used
160d163575Sopenharmony_ci *    to endorse or promote products derived from this software without specific prior written
170d163575Sopenharmony_ci *    permission.
180d163575Sopenharmony_ci *
190d163575Sopenharmony_ci * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
200d163575Sopenharmony_ci * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
210d163575Sopenharmony_ci * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
220d163575Sopenharmony_ci * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
230d163575Sopenharmony_ci * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
240d163575Sopenharmony_ci * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
250d163575Sopenharmony_ci * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
260d163575Sopenharmony_ci * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
270d163575Sopenharmony_ci * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
280d163575Sopenharmony_ci * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
290d163575Sopenharmony_ci * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
300d163575Sopenharmony_ci */
310d163575Sopenharmony_ci
320d163575Sopenharmony_ci#include "pthread.h"
330d163575Sopenharmony_ci#include "pprivate.h"
340d163575Sopenharmony_ci
350d163575Sopenharmony_ci
360d163575Sopenharmony_ciint pthread_attr_init(pthread_attr_t *attr)
370d163575Sopenharmony_ci{
380d163575Sopenharmony_ci    if (attr == NULL) {
390d163575Sopenharmony_ci        return EINVAL;
400d163575Sopenharmony_ci    }
410d163575Sopenharmony_ci
420d163575Sopenharmony_ci    attr->detachstate                 = PTHREAD_CREATE_JOINABLE;
430d163575Sopenharmony_ci    attr->schedpolicy                 = SCHED_RR;
440d163575Sopenharmony_ci    attr->schedparam.sched_priority   = LOSCFG_BASE_CORE_TSK_DEFAULT_PRIO;
450d163575Sopenharmony_ci    attr->inheritsched                = PTHREAD_INHERIT_SCHED;
460d163575Sopenharmony_ci    attr->scope                       = PTHREAD_SCOPE_PROCESS;
470d163575Sopenharmony_ci    attr->stackaddr_set               = 0;
480d163575Sopenharmony_ci    attr->stackaddr                   = NULL;
490d163575Sopenharmony_ci    attr->stacksize_set               = 1;
500d163575Sopenharmony_ci    attr->stacksize                   = LOSCFG_BASE_CORE_TSK_DEFAULT_STACK_SIZE;
510d163575Sopenharmony_ci
520d163575Sopenharmony_ci#ifdef LOSCFG_KERNEL_SMP
530d163575Sopenharmony_ci    attr->cpuset.__bits[0] = 0;
540d163575Sopenharmony_ci#endif
550d163575Sopenharmony_ci
560d163575Sopenharmony_ci    return ENOERR;
570d163575Sopenharmony_ci}
580d163575Sopenharmony_ci
590d163575Sopenharmony_ciint pthread_attr_destroy(pthread_attr_t *attr)
600d163575Sopenharmony_ci{
610d163575Sopenharmony_ci    if (attr == NULL) {
620d163575Sopenharmony_ci        return EINVAL;
630d163575Sopenharmony_ci    }
640d163575Sopenharmony_ci
650d163575Sopenharmony_ci    /* Nothing to do here... */
660d163575Sopenharmony_ci    return ENOERR;
670d163575Sopenharmony_ci}
680d163575Sopenharmony_ci
690d163575Sopenharmony_ciint pthread_attr_setdetachstate(pthread_attr_t *attr, int detachState)
700d163575Sopenharmony_ci{
710d163575Sopenharmony_ci    if ((attr != NULL) && ((detachState == PTHREAD_CREATE_JOINABLE) || (detachState == PTHREAD_CREATE_DETACHED))) {
720d163575Sopenharmony_ci        attr->detachstate = (UINT32)detachState;
730d163575Sopenharmony_ci        return ENOERR;
740d163575Sopenharmony_ci    }
750d163575Sopenharmony_ci
760d163575Sopenharmony_ci    return EINVAL;
770d163575Sopenharmony_ci}
780d163575Sopenharmony_ci
790d163575Sopenharmony_ciint pthread_attr_getdetachstate(const pthread_attr_t *attr, int *detachState)
800d163575Sopenharmony_ci{
810d163575Sopenharmony_ci    if ((attr == NULL) || (detachState == NULL)) {
820d163575Sopenharmony_ci        return EINVAL;
830d163575Sopenharmony_ci    }
840d163575Sopenharmony_ci
850d163575Sopenharmony_ci    *detachState = (int)attr->detachstate;
860d163575Sopenharmony_ci
870d163575Sopenharmony_ci    return ENOERR;
880d163575Sopenharmony_ci}
890d163575Sopenharmony_ci
900d163575Sopenharmony_ciint pthread_attr_setscope(pthread_attr_t *attr, int scope)
910d163575Sopenharmony_ci{
920d163575Sopenharmony_ci    if (attr == NULL) {
930d163575Sopenharmony_ci        return EINVAL;
940d163575Sopenharmony_ci    }
950d163575Sopenharmony_ci
960d163575Sopenharmony_ci    if (scope == PTHREAD_SCOPE_PROCESS) {
970d163575Sopenharmony_ci        attr->scope = (unsigned int)scope;
980d163575Sopenharmony_ci        return ENOERR;
990d163575Sopenharmony_ci    }
1000d163575Sopenharmony_ci
1010d163575Sopenharmony_ci    if (scope == PTHREAD_SCOPE_SYSTEM) {
1020d163575Sopenharmony_ci        return ENOTSUP;
1030d163575Sopenharmony_ci    }
1040d163575Sopenharmony_ci
1050d163575Sopenharmony_ci    return EINVAL;
1060d163575Sopenharmony_ci}
1070d163575Sopenharmony_ci
1080d163575Sopenharmony_ciint pthread_attr_getscope(const pthread_attr_t *attr, int *scope)
1090d163575Sopenharmony_ci{
1100d163575Sopenharmony_ci    if ((attr == NULL) || (scope == NULL)) {
1110d163575Sopenharmony_ci        return EINVAL;
1120d163575Sopenharmony_ci    }
1130d163575Sopenharmony_ci
1140d163575Sopenharmony_ci    *scope = (int)attr->scope;
1150d163575Sopenharmony_ci
1160d163575Sopenharmony_ci    return ENOERR;
1170d163575Sopenharmony_ci}
1180d163575Sopenharmony_ci
1190d163575Sopenharmony_ciint pthread_attr_setinheritsched(pthread_attr_t *attr, int inherit)
1200d163575Sopenharmony_ci{
1210d163575Sopenharmony_ci    if ((attr != NULL) && ((inherit == PTHREAD_INHERIT_SCHED) || (inherit == PTHREAD_EXPLICIT_SCHED))) {
1220d163575Sopenharmony_ci        attr->inheritsched = (UINT32)inherit;
1230d163575Sopenharmony_ci        return ENOERR;
1240d163575Sopenharmony_ci    }
1250d163575Sopenharmony_ci
1260d163575Sopenharmony_ci    return EINVAL;
1270d163575Sopenharmony_ci}
1280d163575Sopenharmony_ci
1290d163575Sopenharmony_ciint pthread_attr_getinheritsched(const pthread_attr_t *attr, int *inherit)
1300d163575Sopenharmony_ci{
1310d163575Sopenharmony_ci    if ((attr == NULL) || (inherit == NULL)) {
1320d163575Sopenharmony_ci        return EINVAL;
1330d163575Sopenharmony_ci    }
1340d163575Sopenharmony_ci
1350d163575Sopenharmony_ci    *inherit = (int)attr->inheritsched;
1360d163575Sopenharmony_ci
1370d163575Sopenharmony_ci    return ENOERR;
1380d163575Sopenharmony_ci}
1390d163575Sopenharmony_ci
1400d163575Sopenharmony_ciint pthread_attr_setschedpolicy(pthread_attr_t *attr, int policy)
1410d163575Sopenharmony_ci{
1420d163575Sopenharmony_ci    if ((attr != NULL) && (policy == SCHED_RR)) {
1430d163575Sopenharmony_ci        attr->schedpolicy = SCHED_RR;
1440d163575Sopenharmony_ci        return ENOERR;
1450d163575Sopenharmony_ci    }
1460d163575Sopenharmony_ci
1470d163575Sopenharmony_ci    return EINVAL;
1480d163575Sopenharmony_ci}
1490d163575Sopenharmony_ci
1500d163575Sopenharmony_ciint pthread_attr_getschedpolicy(const pthread_attr_t *attr, int *policy)
1510d163575Sopenharmony_ci{
1520d163575Sopenharmony_ci    if ((attr == NULL) || (policy == NULL)) {
1530d163575Sopenharmony_ci        return EINVAL;
1540d163575Sopenharmony_ci    }
1550d163575Sopenharmony_ci
1560d163575Sopenharmony_ci    *policy = (int)attr->schedpolicy;
1570d163575Sopenharmony_ci
1580d163575Sopenharmony_ci    return ENOERR;
1590d163575Sopenharmony_ci}
1600d163575Sopenharmony_ci
1610d163575Sopenharmony_ciint pthread_attr_setschedparam(pthread_attr_t *attr, const struct sched_param *param)
1620d163575Sopenharmony_ci{
1630d163575Sopenharmony_ci    if ((attr == NULL) || (param == NULL)) {
1640d163575Sopenharmony_ci        return EINVAL;
1650d163575Sopenharmony_ci    } else if ((param->sched_priority < 0) || (param->sched_priority > OS_TASK_PRIORITY_LOWEST)) {
1660d163575Sopenharmony_ci        return ENOTSUP;
1670d163575Sopenharmony_ci    }
1680d163575Sopenharmony_ci
1690d163575Sopenharmony_ci    attr->schedparam = *param;
1700d163575Sopenharmony_ci
1710d163575Sopenharmony_ci    return ENOERR;
1720d163575Sopenharmony_ci}
1730d163575Sopenharmony_ci
1740d163575Sopenharmony_ciint pthread_attr_getschedparam(const pthread_attr_t *attr, struct sched_param *param)
1750d163575Sopenharmony_ci{
1760d163575Sopenharmony_ci    if ((attr == NULL) || (param == NULL)) {
1770d163575Sopenharmony_ci        return EINVAL;
1780d163575Sopenharmony_ci    }
1790d163575Sopenharmony_ci
1800d163575Sopenharmony_ci    *param = attr->schedparam;
1810d163575Sopenharmony_ci
1820d163575Sopenharmony_ci    return ENOERR;
1830d163575Sopenharmony_ci}
1840d163575Sopenharmony_ci
1850d163575Sopenharmony_ci/*
1860d163575Sopenharmony_ci * Set starting address of stack. Whether this is at the start or end of
1870d163575Sopenharmony_ci * the memory block allocated for the stack depends on whether the stack
1880d163575Sopenharmony_ci * grows up or down.
1890d163575Sopenharmony_ci */
1900d163575Sopenharmony_ciint pthread_attr_setstackaddr(pthread_attr_t *attr, void *stackAddr)
1910d163575Sopenharmony_ci{
1920d163575Sopenharmony_ci    if (attr == NULL) {
1930d163575Sopenharmony_ci        return EINVAL;
1940d163575Sopenharmony_ci    }
1950d163575Sopenharmony_ci
1960d163575Sopenharmony_ci    attr->stackaddr_set = 1;
1970d163575Sopenharmony_ci    attr->stackaddr     = stackAddr;
1980d163575Sopenharmony_ci
1990d163575Sopenharmony_ci    return ENOERR;
2000d163575Sopenharmony_ci}
2010d163575Sopenharmony_ci
2020d163575Sopenharmony_ciint pthread_attr_getstackaddr(const pthread_attr_t *attr, void **stackAddr)
2030d163575Sopenharmony_ci{
2040d163575Sopenharmony_ci    if (((attr != NULL) && (stackAddr != NULL)) && attr->stackaddr_set) {
2050d163575Sopenharmony_ci        *stackAddr = attr->stackaddr;
2060d163575Sopenharmony_ci        return ENOERR;
2070d163575Sopenharmony_ci    }
2080d163575Sopenharmony_ci
2090d163575Sopenharmony_ci    return EINVAL; /* Stack address not set, return EINVAL. */
2100d163575Sopenharmony_ci}
2110d163575Sopenharmony_ci
2120d163575Sopenharmony_ciint pthread_attr_setstacksize(pthread_attr_t *attr, size_t stackSize)
2130d163575Sopenharmony_ci{
2140d163575Sopenharmony_ci    /* Reject inadequate stack sizes */
2150d163575Sopenharmony_ci    if ((attr == NULL) || (stackSize < PTHREAD_STACK_MIN)) {
2160d163575Sopenharmony_ci        return EINVAL;
2170d163575Sopenharmony_ci    }
2180d163575Sopenharmony_ci
2190d163575Sopenharmony_ci    attr->stacksize_set = 1;
2200d163575Sopenharmony_ci    attr->stacksize     = stackSize;
2210d163575Sopenharmony_ci
2220d163575Sopenharmony_ci    return ENOERR;
2230d163575Sopenharmony_ci}
2240d163575Sopenharmony_ci
2250d163575Sopenharmony_ciint pthread_attr_getstacksize(const pthread_attr_t *attr, size_t *stackSize)
2260d163575Sopenharmony_ci{
2270d163575Sopenharmony_ci    /* Reject attempts to get a stack size when one has not been set. */
2280d163575Sopenharmony_ci    if ((attr == NULL) || (stackSize == NULL) || (!attr->stacksize_set)) {
2290d163575Sopenharmony_ci        return EINVAL;
2300d163575Sopenharmony_ci    }
2310d163575Sopenharmony_ci
2320d163575Sopenharmony_ci    *stackSize = attr->stacksize;
2330d163575Sopenharmony_ci
2340d163575Sopenharmony_ci    return ENOERR;
2350d163575Sopenharmony_ci}
2360d163575Sopenharmony_ci
2370d163575Sopenharmony_ci/*
2380d163575Sopenharmony_ci * Set the cpu affinity mask
2390d163575Sopenharmony_ci */
2400d163575Sopenharmony_ciint pthread_attr_setaffinity_np(pthread_attr_t* attr, size_t cpusetsize, const cpu_set_t* cpuset)
2410d163575Sopenharmony_ci{
2420d163575Sopenharmony_ci#ifdef LOSCFG_KERNEL_SMP
2430d163575Sopenharmony_ci    if (attr == NULL) {
2440d163575Sopenharmony_ci        return EINVAL;
2450d163575Sopenharmony_ci    }
2460d163575Sopenharmony_ci
2470d163575Sopenharmony_ci    if ((cpuset == NULL) || (cpusetsize == 0)) {
2480d163575Sopenharmony_ci        attr->cpuset.__bits[0] = 0;
2490d163575Sopenharmony_ci        return ENOERR;
2500d163575Sopenharmony_ci    }
2510d163575Sopenharmony_ci
2520d163575Sopenharmony_ci    if ((cpusetsize != sizeof(cpu_set_t)) || (cpuset->__bits[0] > LOSCFG_KERNEL_CPU_MASK)) {
2530d163575Sopenharmony_ci        return EINVAL;
2540d163575Sopenharmony_ci    }
2550d163575Sopenharmony_ci
2560d163575Sopenharmony_ci    attr->cpuset = *cpuset;
2570d163575Sopenharmony_ci#endif
2580d163575Sopenharmony_ci
2590d163575Sopenharmony_ci    return ENOERR;
2600d163575Sopenharmony_ci}
2610d163575Sopenharmony_ci
2620d163575Sopenharmony_ci/*
2630d163575Sopenharmony_ci * Get the cpu affinity mask
2640d163575Sopenharmony_ci */
2650d163575Sopenharmony_ciint pthread_attr_getaffinity_np(const pthread_attr_t* attr, size_t cpusetsize, cpu_set_t* cpuset)
2660d163575Sopenharmony_ci{
2670d163575Sopenharmony_ci#ifdef LOSCFG_KERNEL_SMP
2680d163575Sopenharmony_ci    if ((attr == NULL) || (cpuset == NULL) || (cpusetsize != sizeof(cpu_set_t))) {
2690d163575Sopenharmony_ci        return EINVAL;
2700d163575Sopenharmony_ci    }
2710d163575Sopenharmony_ci
2720d163575Sopenharmony_ci    *cpuset = attr->cpuset;
2730d163575Sopenharmony_ci#endif
2740d163575Sopenharmony_ci
2750d163575Sopenharmony_ci    return ENOERR;
2760d163575Sopenharmony_ci}
2770d163575Sopenharmony_ci
278