/**************************************************************************** * libs/libc/stdlib/lib_atexit.c * * 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 /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ /**************************************************************************** * Private Functions ****************************************************************************/ /**************************************************************************** * Name: get_exitfuncs * * Description: * Obtain the list of exit functions. * * Input Parameters: * None * * Returned Value: * Pointer to the list of exit functions. * ****************************************************************************/ static FAR struct atexit_list_s * get_exitfuncs(void) { FAR struct task_info_s *info; info = task_get_info(); return &info->ta_exit; } /**************************************************************************** * Name: exitfunc_lock * * Description: * Obtain the exit function lock. * * Returned Value: * OK on success, or negated errno on failure * ****************************************************************************/ static int exitfunc_lock(void) { FAR struct task_info_s *info = task_get_info(); int ret = _SEM_WAIT(&info->ta_sem); if (ret < 0) { ret = _SEM_ERRVAL(ret); } return ret; } /**************************************************************************** * Name: exitfunc_unlock * * Description: * Release exit function lock . * ****************************************************************************/ static void exitfunc_unlock(void) { FAR struct task_info_s *info = task_get_info(); _SEM_POST(&info->ta_sem); } /**************************************************************************** * Public Functions ****************************************************************************/ /**************************************************************************** * Name: atexit * * Description: * Registers a function to be called at program exit. * The atexit() function registers the given function to be called * at normal process termination, whether via exit or via return from * the program's main(). * * Limitations in the current implementation: * * 1. Only a single atexit function can be registered unless * CONFIG_LIBC_MAX_EXITFUNS defines a larger number. * 2. atexit functions are not inherited when a new task is * created. * * Input Parameters: * func - A pointer to the function to be called when the task exits. * * Returned Value: * Zero on success. Non-zero on failure. * ****************************************************************************/ int atexit(CODE void (*func)(void)) { return atexit_register(ATTYPE_ATEXIT, func, NULL, NULL); } int atexit_register(int type, CODE void (*func)(void), FAR void *arg, FAR void *dso) { FAR struct atexit_list_s *aehead; int idx; int ret = ERROR; /* REVISIT: Missing logic */ UNUSED(dso); /* The following must be atomic */ aehead = get_exitfuncs(); if (func) { ret = exitfunc_lock(); if (ret < 0) { return ret; } if ((idx = aehead->nfuncs) < ATEXIT_MAX) { aehead->funcs[idx].type = type; aehead->funcs[idx].func = func; aehead->funcs[idx].arg = arg; aehead->nfuncs++; ret = OK; } else { ret = ERROR; } exitfunc_unlock(); } return ret; } void atexit_call_exitfuncs(int status) { FAR struct atexit_list_s *aehead; CODE void (*func)(void); FAR void *arg; int idx; int type; /* Call exit functions in reverse order */ aehead = get_exitfuncs(); for (idx = aehead->nfuncs - 1; idx >= 0; idx--) { /* Remove the function to prevent recursive call to it */ type = aehead->funcs[idx].type; func = aehead->funcs[idx].func; arg = aehead->funcs[idx].arg; aehead->funcs[idx].func = NULL; aehead->funcs[idx].arg = NULL; if (!func) { continue; } /* Call the atexit/on_exit/cxa_atexit() function */ if (type == ATTYPE_ATEXIT) { (*func)(); } else if (type == ATTYPE_ONEXIT) { (*((CODE void (*)(int, FAR void *))func))(status, arg); } else if (type == ATTYPE_CXA) { (*((CODE void (*)(FAR void *))func))(arg); } } }