/**************************************************************************** * libs/libc/sched/task_setcancelstate.c * * SPDX-License-Identifier: Apache-2.0 * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. The * ASF licenses this file to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance with the * License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations * under the License. * ****************************************************************************/ /**************************************************************************** * Included Files ****************************************************************************/ #include #include #include #include #include #include #include #include /**************************************************************************** * Public Functions ****************************************************************************/ /**************************************************************************** * Name: task_setcancelstate * * Description: * The task_setcancelstate() function atomically both sets the calling * task's cancellability state to the indicated state and returns the * previous cancellability state at the location referenced by oldstate. * Legal values for state are TASK_CANCEL_ENABLE and TASK_CANCEL_DISABLE. * * The cancellability state and type of any newly created tasks are * TASK_CANCEL_ENABLE and TASK_CANCEL_DEFERRED respectively. * * Input Parameters: * state - the new cancellability state, either TASK_CANCEL_ENABLE or * TASK_CANCEL_DISABLE * oldstate - The location to return the old cancellability state. * * Returned Value: * Zero (OK) on success; ERROR is returned on any failure with the * errno value set appropriately. * ****************************************************************************/ int task_setcancelstate(int state, FAR int *oldstate) { FAR struct tls_info_s *tls = tls_get_info(); int ret = OK; /* Return the current state if so requested */ if (oldstate != NULL) { if ((tls->tl_cpstate & CANCEL_FLAG_NONCANCELABLE) != 0) { *oldstate = TASK_CANCEL_DISABLE; } else { *oldstate = TASK_CANCEL_ENABLE; } } /* Set the new cancellation state */ if (state == TASK_CANCEL_ENABLE) { /* Clear the non-cancelable flag */ tls->tl_cpstate &= ~CANCEL_FLAG_NONCANCELABLE; /* Check if a cancellation was pending */ if ((tls->tl_cpstate & CANCEL_FLAG_CANCEL_PENDING) != 0) { #ifdef CONFIG_CANCELLATION_POINTS /* If we are using deferred cancellation? */ if ((tls->tl_cpstate & CANCEL_FLAG_CANCEL_ASYNC) != 0) #endif { /* No.. We are using asynchronous cancellation. If the * cancellation was pending in this case, then just exit. */ tls->tl_cpstate &= ~CANCEL_FLAG_CANCEL_PENDING; #ifndef CONFIG_DISABLE_PTHREAD pthread_exit(PTHREAD_CANCELED); #else exit(EXIT_FAILURE); #endif } } } else if (state == TASK_CANCEL_DISABLE) { /* Set the non-cancelable state */ tls->tl_cpstate |= CANCEL_FLAG_NONCANCELABLE; } else { set_errno(EINVAL); ret = ERROR; } return ret; }