mirror of
https://github.com/adtools/clib2.git
synced 2025-12-08 14:59:05 +00:00
into a dedicated module. This also has the advantage that it's harder to break code by accidentally forgetting to call InitSemaphore() after having allocated the memory for it. git-svn-id: file:///Users/olsen/Code/migration-svn-zu-git/logical-line-staging/clib2/trunk@14853 87f5fb63-7c3d-0410-a384-fd976d0f7a62
314 lines
7.6 KiB
C
314 lines
7.6 KiB
C
/*
|
|
* $Id: socket_init_exit.c,v 1.11 2005-03-03 14:20:55 obarthel Exp $
|
|
*
|
|
* :ts=4
|
|
*
|
|
* Portable ISO 'C' (1994) runtime library for the Amiga computer
|
|
* Copyright (c) 2002-2005 by Olaf Barthel <olsen@sourcery.han.de>
|
|
* 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.
|
|
*/
|
|
|
|
#if defined(SOCKET_SUPPORT)
|
|
|
|
/****************************************************************************/
|
|
|
|
#ifndef _SOCKET_HEADERS_H
|
|
#include "socket_headers.h"
|
|
#endif /* _SOCKET_HEADERS_H */
|
|
|
|
#ifndef _SIGNAL_HEADERS_H
|
|
#include "signal_headers.h"
|
|
#endif /* _SIGNAL_HEADERS_H */
|
|
|
|
/****************************************************************************/
|
|
|
|
#include "stdlib_protos.h"
|
|
|
|
/****************************************************************************/
|
|
|
|
/* Code value. */
|
|
#define SBTB_CODE 1
|
|
#define SBTS_CODE 0x3FFF
|
|
|
|
/* Set a parameter, passing it by value. */
|
|
#ifndef SBTM_SETVAL
|
|
#define SBTM_SETVAL(code) (TAG_USER | (((code) & SBTS_CODE) << SBTB_CODE) | 1)
|
|
#endif /* SBTM_SETVAL */
|
|
|
|
#define SBTC_BREAKMASK 1 /* Interrupt signal mask */
|
|
#define SBTC_LOGTAGPTR 11 /* Under which name log entries are filed */
|
|
#define SBTC_ERRNOLONGPTR 24 /* Pointer to errno, length == sizeof(errno) */
|
|
#define SBTC_HERRNOLONGPTR 25 /* 'h_errno' pointer (with sizeof(h_errno) == sizeof(long)) */
|
|
|
|
/****************************************************************************/
|
|
|
|
struct DaemonMessage
|
|
{
|
|
struct Message dm_Message;
|
|
ULONG dm_Pad1;
|
|
ULONG dm_Pad2;
|
|
LONG dm_ID;
|
|
ULONG dm_Pad3;
|
|
UBYTE dm_Family;
|
|
UBYTE dm_Type;
|
|
};
|
|
|
|
/****************************************************************************/
|
|
|
|
CLIB_DESTRUCTOR(__socket_exit)
|
|
{
|
|
ENTER();
|
|
|
|
/* Disable ^C checking. */
|
|
if(__SocketBase != NULL)
|
|
{
|
|
struct TagItem tags[2];
|
|
|
|
tags[0].ti_Tag = SBTM_SETVAL(SBTC_BREAKMASK);
|
|
tags[0].ti_Data = 0;
|
|
tags[1].ti_Tag = TAG_END;
|
|
|
|
PROFILE_OFF();
|
|
__SocketBaseTagList(tags);
|
|
PROFILE_ON();
|
|
}
|
|
|
|
/* Careful: if this function is ever invoked, it must make sure that
|
|
* the socket file descriptors are invalidated. If that
|
|
* does not happen, the stdio cleanup function will
|
|
* crash (with bells on).
|
|
*/
|
|
__close_all_files();
|
|
|
|
#if defined(__amigaos4__)
|
|
{
|
|
if(__ISocket != NULL)
|
|
{
|
|
DropInterface((struct Interface *)__ISocket);
|
|
__ISocket = NULL;
|
|
}
|
|
}
|
|
#endif /* __amigaos4__ */
|
|
|
|
if(__SocketBase != NULL)
|
|
{
|
|
CloseLibrary(__SocketBase);
|
|
__SocketBase = NULL;
|
|
}
|
|
|
|
LEAVE();
|
|
}
|
|
|
|
/****************************************************************************/
|
|
|
|
int
|
|
__socket_init(void)
|
|
{
|
|
struct TagItem tags[5];
|
|
int result = ERROR;
|
|
LONG status;
|
|
|
|
ENTER();
|
|
|
|
PROFILE_OFF();
|
|
|
|
/* bsdsocket.library V3 is sufficient for all the tasks we
|
|
* may have to perform.
|
|
*/
|
|
__SocketBase = OpenLibrary("bsdsocket.library",3);
|
|
|
|
#if defined(__amigaos4__)
|
|
{
|
|
if(__SocketBase != NULL)
|
|
{
|
|
__ISocket = (struct SocketIFace *)GetInterface(__SocketBase, "main", 1, 0);
|
|
if (__ISocket == NULL)
|
|
{
|
|
CloseLibrary(__SocketBase);
|
|
__SocketBase = NULL;
|
|
}
|
|
}
|
|
}
|
|
#endif /* __amigaos4__ */
|
|
|
|
PROFILE_ON();
|
|
|
|
if(__SocketBase == NULL)
|
|
{
|
|
SHOWMSG("bsdsocket.library V3 didn't open");
|
|
|
|
__show_error("\"bsdsocket.library\" V3 could not be opened.");
|
|
goto out;
|
|
}
|
|
|
|
/* Wire the library's errno variable to our local errno. */
|
|
tags[0].ti_Tag = SBTM_SETVAL(SBTC_ERRNOLONGPTR);
|
|
tags[0].ti_Data = (ULONG)&errno;
|
|
|
|
/* Also enable ^C checking if desired. */
|
|
tags[1].ti_Tag = SBTM_SETVAL(SBTC_BREAKMASK);
|
|
|
|
if(__check_abort_enabled)
|
|
tags[1].ti_Data = SIGBREAKF_CTRL_C;
|
|
else
|
|
tags[1].ti_Data = 0;
|
|
|
|
tags[2].ti_Tag = SBTM_SETVAL(SBTC_LOGTAGPTR);
|
|
tags[2].ti_Data = (ULONG)__program_name;
|
|
|
|
/* Wire the library's h_errno variable to our local h_errno. */
|
|
tags[3].ti_Tag = SBTM_SETVAL(SBTC_HERRNOLONGPTR);
|
|
tags[3].ti_Data = (ULONG)&h_errno;
|
|
|
|
tags[4].ti_Tag = TAG_END;
|
|
|
|
PROFILE_OFF();
|
|
status = __SocketBaseTagList(tags);
|
|
PROFILE_ON();
|
|
|
|
if(status != 0)
|
|
{
|
|
SHOWMSG("couldn't initialize the library");
|
|
|
|
__show_error("\"bsdsocket.library\" could not be initialized.");
|
|
goto out;
|
|
}
|
|
|
|
/* Check if this program was launched as a server by the Internet
|
|
* superserver.
|
|
*/
|
|
if(Cli() != NULL && NOT __detach)
|
|
{
|
|
struct DaemonMessage * dm;
|
|
|
|
/* The socket the superserver may have launched this program
|
|
* with is attached to the exit data entry of the process.
|
|
*/
|
|
#if defined(__amigaos4__)
|
|
{
|
|
dm = (struct DaemonMessage *)GetExitData();
|
|
}
|
|
#else
|
|
{
|
|
struct Process * this_process = (struct Process *)FindTask(NULL);
|
|
|
|
dm = (struct DaemonMessage *)this_process->pr_ExitData;
|
|
}
|
|
#endif /* __amigaos4__ */
|
|
|
|
if(TypeOfMem(dm) != 0 && TypeOfMem(((char *)dm) + sizeof(*dm)-1) != 0)
|
|
{
|
|
struct SignalSemaphore * lock;
|
|
int daemon_socket;
|
|
struct fd * fd;
|
|
int sockfd;
|
|
int i;
|
|
|
|
SHOWMSG("we have a daemon message; this is a server");
|
|
|
|
/* Try to grab that socket and attach is to the three
|
|
* standard I/O streams.
|
|
*/
|
|
|
|
PROFILE_OFF();
|
|
daemon_socket = __ObtainSocket(dm->dm_ID,dm->dm_Family,dm->dm_Type,0);
|
|
PROFILE_ON();
|
|
|
|
if(daemon_socket == -1)
|
|
{
|
|
__show_error("Network server streams could not be initialized.");
|
|
goto out;
|
|
}
|
|
|
|
SHOWVALUE(daemon_socket);
|
|
|
|
/* Shut down the three standard I/O streams. */
|
|
for(i = STDIN_FILENO ; i <= STDERR_FILENO ; i++)
|
|
close(i);
|
|
|
|
/* The standard I/O streams are no longer attached to a console. */
|
|
__no_standard_io = TRUE;
|
|
|
|
/* Put the socket into the three standard I/O streams. */
|
|
for(i = STDIN_FILENO ; i <= STDERR_FILENO ; i++)
|
|
{
|
|
#if defined(__THREAD_SAFE)
|
|
{
|
|
lock = __create_semaphore();
|
|
if(lock == NULL)
|
|
goto out;
|
|
}
|
|
#else
|
|
{
|
|
lock = NULL;
|
|
}
|
|
#endif /* __THREAD_SAFE */
|
|
|
|
fd = __fd[i];
|
|
|
|
assert( fd != NULL && FLAG_IS_CLEAR(fd->fd_Flags,FDF_IN_USE) );
|
|
|
|
if(i == STDIN_FILENO)
|
|
{
|
|
sockfd = daemon_socket;
|
|
}
|
|
else
|
|
{
|
|
PROFILE_OFF();
|
|
sockfd = __Dup2Socket(daemon_socket,-1);
|
|
PROFILE_ON();
|
|
|
|
if(sockfd == -1)
|
|
{
|
|
SHOWMSG("could not duplicate daemon socket");
|
|
|
|
__delete_semaphore(lock);
|
|
|
|
__show_error("Network server streams could not be initialized.");
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
__initialize_fd(fd,__socket_hook_entry,(BPTR)sockfd,FDF_IN_USE | FDF_IS_SOCKET | FDF_READ | FDF_WRITE,lock);
|
|
}
|
|
|
|
/* This program now runs as an internet superserver client (daemon). */
|
|
__is_daemon = TRUE;
|
|
}
|
|
}
|
|
|
|
result = OK;
|
|
|
|
out:
|
|
|
|
RETURN(result);
|
|
return(result);
|
|
}
|
|
|
|
/****************************************************************************/
|
|
|
|
#endif /* SOCKET_SUPPORT */
|