/* * $Id: stdio_init_exit.c,v 1.2 2004-09-29 14:17:44 obarthel Exp $ * * :ts=4 * * Portable ISO 'C' (1994) runtime library for the Amiga computer * Copyright (c) 2002-2004 by Olaf Barthel * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * - Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * - Neither the name of Olaf Barthel nor the names of contributors * may be used to endorse or promote products derived from this * software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #ifndef _STDLIB_MEM_DEBUG_H #include "stdlib_mem_debug.h" #endif /* _STDLIB_MEM_DEBUG_H */ /****************************************************************************/ #ifndef _STDIO_HEADERS_H #include "stdio_headers.h" #endif /* _STDIO_HEADERS_H */ #ifndef _UNISTD_HEADERS_H #include "unistd_headers.h" #endif /* _UNISTD_HEADERS_H */ /****************************************************************************/ #include "stdlib_protos.h" /****************************************************************************/ void __close_all_files(void) { int i; ENTER(); __check_abort_enabled = FALSE; if(__iob != NULL && __num_iob > 0) { for(i = 0 ; i < __num_iob ; i++) { if(FLAG_IS_SET(__iob[i]->iob_Flags,IOBF_IN_USE)) fclose((FILE *)__iob[i]); } __num_iob = 0; __iob = NULL; } if(__fd != NULL && __num_fd > 0) { for(i = 0 ; i < __num_fd ; i++) { if(FLAG_IS_SET(__fd[i]->fd_Flags,FDF_IN_USE)) close(i); } __num_fd = 0; __fd = NULL; } LEAVE(); } /****************************************************************************/ CLIB_DESTRUCTOR(__stdio_exit) { ENTER(); __close_all_files(); LEAVE(); } /****************************************************************************/ int __stdio_init(void) { const int num_standard_files = 3; BPTR default_file; ULONG fd_flags,iob_flags; int result = ERROR; char * buffer; char * aligned_buffer; int i; ENTER(); __iob = malloc(sizeof(*__iob) * num_standard_files); if(__iob == NULL) goto out; for(i = 0 ; i < num_standard_files ; i++) { __iob[i] = malloc(sizeof(*__iob[i])); if(__iob[i] == NULL) goto out; memset(__iob[i],0,sizeof(*__iob[i])); } __num_iob = num_standard_files; __fd = malloc(sizeof(*__fd) * num_standard_files); if(__fd == NULL) goto out; for(i = 0 ; i < num_standard_files ; i++) { __fd[i] = malloc(sizeof(*__fd[i])); if(__fd[i] == NULL) goto out; memset(__fd[i],0,sizeof(*__fd[i])); } __num_fd = num_standard_files; /* Now initialize the standard I/O streams (input, output, error). */ for(i = STDIN_FILENO ; i <= STDERR_FILENO ; i++) { PROFILE_OFF(); switch(i) { case STDIN_FILENO: iob_flags = IOBF_IN_USE | IOBF_READ | IOBF_NO_NUL | IOBF_BUFFER_MODE_LINE; fd_flags = FDF_IN_USE | FDF_READ | FDF_NO_CLOSE; default_file = Input(); break; case STDOUT_FILENO: iob_flags = IOBF_IN_USE | IOBF_WRITE | IOBF_NO_NUL | IOBF_BUFFER_MODE_LINE; fd_flags = FDF_IN_USE | FDF_WRITE | FDF_NO_CLOSE; default_file = Output(); break; case STDERR_FILENO: default: iob_flags = IOBF_IN_USE | IOBF_WRITE | IOBF_NO_NUL | IOBF_BUFFER_MODE_NONE; fd_flags = FDF_IN_USE | FDF_WRITE; default_file = ZERO; break; } PROFILE_ON(); /* Allocate a little more memory than necessary. */ buffer = malloc(BUFSIZ + 15); if(buffer == NULL) goto out; /* Align the buffer start address to a cache line boundary. */ aligned_buffer = (char *)((ULONG)(buffer + 15) & ~15UL); __initialize_fd(__fd[i],(HOOKFUNC)__fd_hook_entry,default_file,fd_flags); __initialize_iob(__iob[i],(HOOKFUNC)__iob_hook_entry, buffer, aligned_buffer,BUFSIZ, i, i, iob_flags); } /* If the program was launched from Workbench, we continue by * duplicating the default output stream for use as the * standard error stream. */ if(__WBenchMsg != NULL) { PROFILE_OFF(); __fd[STDERR_FILENO]->fd_DefaultFile = Output(); PROFILE_ON(); SET_FLAG(__fd[STDERR_FILENO]->fd_Flags,FDF_NO_CLOSE); } else { struct Process * this_process = (struct Process *)FindTask(NULL); /* Is the standard error stream configured? If so, use it. * Otherwise, try to duplicate the standard output stream. */ if(this_process->pr_CES != ZERO) { __fd[STDERR_FILENO]->fd_DefaultFile = this_process->pr_CES; SET_FLAG(__fd[STDERR_FILENO]->fd_Flags,FDF_NO_CLOSE); } else if (__fd[STDOUT_FILENO]->fd_DefaultFile != ZERO) { struct FileHandle * fh = (struct FileHandle *)BADDR(__fd[STDOUT_FILENO]->fd_DefaultFile); /* Careful, this could be "NIL:". */ if(fh->fh_Type != NULL) __fd[STDERR_FILENO]->fd_DefaultFile = Open("CONSOLE:",MODE_NEWFILE); } } PROFILE_OFF(); /* Check if the standard error output refers to a console or must * be considered unusable for console output. */ if(__fd[STDERR_FILENO]->fd_DefaultFile == ZERO || ((struct FileHandle *)BADDR(__fd[STDERR_FILENO]->fd_DefaultFile))->fh_Type == NULL || NOT IsInteractive(__fd[STDERR_FILENO]->fd_DefaultFile)) { /* The standard I/O streams are no longer attached to a console. */ __no_standard_io = TRUE; } PROFILE_ON(); result = OK; out: RETURN(result); return(result); }