2000-02-18 03:39:52 +08:00
|
|
|
/*
|
|
|
|
* Copyright (c) 1990 Regents of the University of California.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* %sccs.include.redist.c%
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
FUNCTION
|
|
|
|
<<exit>>---end program execution
|
|
|
|
|
|
|
|
INDEX
|
|
|
|
exit
|
|
|
|
|
|
|
|
ANSI_SYNOPSIS
|
|
|
|
#include <stdlib.h>
|
|
|
|
void exit(int <[code]>);
|
|
|
|
|
|
|
|
TRAD_SYNOPSIS
|
|
|
|
#include <stdlib.h>
|
|
|
|
void exit(<[code]>)
|
|
|
|
int <[code]>;
|
|
|
|
|
|
|
|
DESCRIPTION
|
|
|
|
Use <<exit>> to return control from a program to the host operating
|
|
|
|
environment. Use the argument <[code]> to pass an exit status to the
|
|
|
|
operating environment: two particular values, <<EXIT_SUCCESS>> and
|
|
|
|
<<EXIT_FAILURE>>, are defined in `<<stdlib.h>>' to indicate success or
|
|
|
|
failure in a portable fashion.
|
|
|
|
|
|
|
|
<<exit>> does two kinds of cleanup before ending execution of your
|
|
|
|
program. First, it calls all application-defined cleanup functions
|
|
|
|
you have enrolled with <<atexit>>. Second, files and streams are
|
|
|
|
cleaned up: any pending output is delivered to the host system, each
|
|
|
|
open file or stream is closed, and files created by <<tmpfile>> are
|
|
|
|
deleted.
|
|
|
|
|
|
|
|
RETURNS
|
|
|
|
<<exit>> does not return to its caller.
|
|
|
|
|
|
|
|
PORTABILITY
|
|
|
|
ANSI C requires <<exit>>, and specifies that <<EXIT_SUCCESS>> and
|
|
|
|
<<EXIT_FAILURE>> must be defined.
|
|
|
|
|
|
|
|
Supporting OS subroutines required: <<_exit>>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <unistd.h> /* for _exit() declaration */
|
|
|
|
#include <reent.h>
|
|
|
|
|
|
|
|
#ifndef _REENT_ONLY
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Exit, flushing stdio buffers if necessary.
|
|
|
|
*/
|
|
|
|
|
|
|
|
void
|
|
|
|
_DEFUN (exit, (code),
|
|
|
|
int code)
|
|
|
|
{
|
|
|
|
register struct _atexit *p;
|
2003-06-06 23:36:31 +08:00
|
|
|
register struct _on_exit_args * args;
|
2000-02-18 03:39:52 +08:00
|
|
|
register int n;
|
2003-06-06 23:36:31 +08:00
|
|
|
int i;
|
|
|
|
|
2002-02-03 17:24:18 +08:00
|
|
|
#ifdef _REENT_SMALL
|
2003-09-06 02:22:00 +08:00
|
|
|
p = &_GLOBAL_REENT->_atexit;
|
2003-06-06 23:36:31 +08:00
|
|
|
args = p->_on_exit_args_ptr;
|
|
|
|
|
|
|
|
if (args == NULL)
|
|
|
|
{
|
|
|
|
for (n = p->_ind; n--;)
|
|
|
|
p->_fns[n] ();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for (n = p->_ind - 1, i = (n >= 0) ? (1 << n) : 0; n >= 0; --n, i >>= 1)
|
|
|
|
if (args->_fntypes & i)
|
|
|
|
(*((void (*)(int, void *)) p->_fns[n]))(code, args->_fnargs[n]);
|
|
|
|
else
|
|
|
|
p->_fns[n] ();
|
|
|
|
}
|
2002-02-03 17:24:18 +08:00
|
|
|
#else
|
2003-09-06 02:22:00 +08:00
|
|
|
p = _GLOBAL_REENT->_atexit;
|
2003-06-06 23:36:31 +08:00
|
|
|
do
|
|
|
|
{
|
|
|
|
args = & p->_on_exit_args;
|
|
|
|
|
|
|
|
for (n = p->_ind - 1, i = (n >= 0) ? (1 << n) : 0; n >= 0; --n, i >>= 1)
|
|
|
|
if (args->_fntypes & i)
|
|
|
|
(*((void (*)(int, void *)) p->_fns[n]))(code, args->_fnargs[n]);
|
|
|
|
else
|
|
|
|
p->_fns[n] ();
|
|
|
|
|
|
|
|
p = p->_next;
|
|
|
|
}
|
|
|
|
while (p);
|
2002-02-03 17:24:18 +08:00
|
|
|
#endif
|
2002-05-16 06:58:10 +08:00
|
|
|
|
2003-09-03 06:42:05 +08:00
|
|
|
if (_GLOBAL_REENT->__cleanup)
|
|
|
|
(*_GLOBAL_REENT->__cleanup) (_GLOBAL_REENT);
|
2000-02-18 03:39:52 +08:00
|
|
|
_exit (code);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|