Blame netware/nwlibc.c

Packit Service df60bb
/*********************************************************************
Packit Service df60bb
 * Universal NetWare library stub.                                   *
Packit Service df60bb
 * written by Ulrich Neuman and given to OpenSource copyright-free.  *
Packit Service df60bb
 * version: 1.0                                                      *
Packit Service df60bb
 * $Id$
Packit Service df60bb
 *********************************************************************/
Packit Service df60bb
Packit Service df60bb
#include <errno.h>
Packit Service df60bb
#include <string.h>
Packit Service df60bb
#include <stdlib.h>
Packit Service df60bb
#include <library.h>
Packit Service df60bb
#include <netware.h>
Packit Service df60bb
#include <screen.h>
Packit Service df60bb
#include <nks/thread.h>
Packit Service df60bb
#include <nks/synch.h>
Packit Service df60bb
Packit Service df60bb
Packit Service df60bb
typedef struct {
Packit Service df60bb
	int     _errno;
Packit Service df60bb
	void    *twentybytes;
Packit Service df60bb
} libthreaddata_t;
Packit Service df60bb
Packit Service df60bb
typedef struct {
Packit Service df60bb
	int         x;
Packit Service df60bb
	int         y;
Packit Service df60bb
	int         z;
Packit Service df60bb
	void        *tenbytes;
Packit Service df60bb
	NXKey_t     perthreadkey;   /* if -1, no key obtained... */
Packit Service df60bb
	NXMutex_t   *lock;
Packit Service df60bb
} libdata_t;
Packit Service df60bb
Packit Service df60bb
int         gLibId      = -1;
Packit Service df60bb
void        *gLibHandle = (void *) NULL;
Packit Service df60bb
rtag_t      gAllocTag   = (rtag_t) NULL;
Packit Service df60bb
NXMutex_t   *gLibLock   = (NXMutex_t *) NULL;
Packit Service df60bb
Packit Service df60bb
/* internal library function prototypes... */
Packit Service df60bb
int     DisposeLibraryData ( void * );
Packit Service df60bb
void    DisposeThreadData ( void * );
Packit Service df60bb
int     GetOrSetUpData ( int id, libdata_t **data, libthreaddata_t **threaddata );
Packit Service df60bb
Packit Service df60bb
Packit Service df60bb
int _NonAppStart
Packit Service df60bb
(
Packit Service df60bb
    void        *NLMHandle,
Packit Service df60bb
    void        *errorScreen,
Packit Service df60bb
    const char  *cmdLine,
Packit Service df60bb
    const char  *loadDirPath,
Packit Service df60bb
    size_t      uninitializedDataLength,
Packit Service df60bb
    void        *NLMFileHandle,
Packit Service df60bb
    int         (*readRoutineP)( int conn, void *fileHandle, size_t offset,
Packit Service df60bb
                                 size_t nbytes, size_t *bytesRead, void *buffer ),
Packit Service df60bb
    size_t      customDataOffset,
Packit Service df60bb
    size_t      customDataSize,
Packit Service df60bb
    int         messageCount,
Packit Service df60bb
    const char  **messages
Packit Service df60bb
)
Packit Service df60bb
{
Packit Service df60bb
	NX_LOCK_INFO_ALLOC(liblock, "Per-Application Data Lock", 0);
Packit Service df60bb
Packit Service df60bb
#ifndef __GNUC__
Packit Service df60bb
#pragma unused(cmdLine)
Packit Service df60bb
#pragma unused(loadDirPath)
Packit Service df60bb
#pragma unused(uninitializedDataLength)
Packit Service df60bb
#pragma unused(NLMFileHandle)
Packit Service df60bb
#pragma unused(readRoutineP)
Packit Service df60bb
#pragma unused(customDataOffset)
Packit Service df60bb
#pragma unused(customDataSize)
Packit Service df60bb
#pragma unused(messageCount)
Packit Service df60bb
#pragma unused(messages)
Packit Service df60bb
#endif
Packit Service df60bb
Packit Service df60bb
	/*
Packit Service df60bb
	** Here we process our command line, post errors (to the error screen),
Packit Service df60bb
	** perform initializations and anything else we need to do before being able
Packit Service df60bb
	** to accept calls into us. If we succeed, we return non-zero and the NetWare
Packit Service df60bb
	** Loader will leave us up, otherwise we fail to load and get dumped.
Packit Service df60bb
	*/
Packit Service df60bb
	gAllocTag = AllocateResourceTag(NLMHandle,
Packit Service df60bb
	                                "<library-name> memory allocations", AllocSignature);
Packit Service df60bb
Packit Service df60bb
	if (!gAllocTag) {
Packit Service df60bb
		OutputToScreen(errorScreen, "Unable to allocate resource tag for "
Packit Service df60bb
		               "library memory allocations.\n");
Packit Service df60bb
		return -1;
Packit Service df60bb
	}
Packit Service df60bb
Packit Service df60bb
	gLibId = register_library(DisposeLibraryData);
Packit Service df60bb
Packit Service df60bb
	if (gLibId < -1) {
Packit Service df60bb
		OutputToScreen(errorScreen, "Unable to register library with kernel.\n");
Packit Service df60bb
		return -1;
Packit Service df60bb
	}
Packit Service df60bb
Packit Service df60bb
	gLibHandle = NLMHandle;
Packit Service df60bb
Packit Service df60bb
	gLibLock = NXMutexAlloc(0, 0, &liblock);
Packit Service df60bb
Packit Service df60bb
	if (!gLibLock) {
Packit Service df60bb
		OutputToScreen(errorScreen, "Unable to allocate library data lock.\n");
Packit Service df60bb
		return -1;
Packit Service df60bb
	}
Packit Service df60bb
Packit Service df60bb
	return 0;
Packit Service df60bb
}
Packit Service df60bb
Packit Service df60bb
/*
Packit Service df60bb
** Here we clean up any resources we allocated. Resource tags is a big part
Packit Service df60bb
** of what we created, but NetWare doesn't ask us to free those.
Packit Service df60bb
*/
Packit Service df60bb
void _NonAppStop( void )
Packit Service df60bb
{
Packit Service df60bb
	(void) unregister_library(gLibId);
Packit Service df60bb
	NXMutexFree(gLibLock);
Packit Service df60bb
}
Packit Service df60bb
Packit Service df60bb
/*
Packit Service df60bb
** This function cannot be the first in the file for if the file is linked
Packit Service df60bb
** first, then the check-unload function's offset will be nlmname.nlm+0
Packit Service df60bb
** which is how to tell that there isn't one. When the check function is
Packit Service df60bb
** first in the linked objects, it is ambiguous. For this reason, we will
Packit Service df60bb
** put it inside this file after the stop function.
Packit Service df60bb
**
Packit Service df60bb
** Here we check to see if it's alright to ourselves to be unloaded. If not,
Packit Service df60bb
** we return a non-zero value. Right now, there isn't any reason not to allow
Packit Service df60bb
** it.
Packit Service df60bb
*/
Packit Service df60bb
int  _NonAppCheckUnload( void )
Packit Service df60bb
{
Packit Service df60bb
	return 0;
Packit Service df60bb
}
Packit Service df60bb
Packit Service df60bb
int GetOrSetUpData
Packit Service df60bb
(
Packit Service df60bb
    int                 id,
Packit Service df60bb
    libdata_t           **appData,
Packit Service df60bb
    libthreaddata_t **threadData
Packit Service df60bb
)
Packit Service df60bb
{
Packit Service df60bb
	int                 err;
Packit Service df60bb
	libdata_t           *app_data;
Packit Service df60bb
	libthreaddata_t *thread_data;
Packit Service df60bb
	NXKey_t             key;
Packit Service df60bb
	NX_LOCK_INFO_ALLOC(liblock, "Application Data Lock", 0);
Packit Service df60bb
Packit Service df60bb
	err         = 0;
Packit Service df60bb
	thread_data = (libthreaddata_t *) NULL;
Packit Service df60bb
Packit Service df60bb
	/*
Packit Service df60bb
	** Attempt to get our data for the application calling us. This is where we
Packit Service df60bb
	** store whatever application-specific information we need to carry in support
Packit Service df60bb
	** of calling applications.
Packit Service df60bb
	*/
Packit Service df60bb
	app_data = (libdata_t *) get_app_data(id);
Packit Service df60bb
Packit Service df60bb
	if (!app_data) {
Packit Service df60bb
		/*
Packit Service df60bb
		** This application hasn't called us before; set up application AND per-thread
Packit Service df60bb
		** data. Of course, just in case a thread from this same application is calling
Packit Service df60bb
		** us simultaneously, we better lock our application data-creation mutex. We
Packit Service df60bb
		** also need to recheck for data after we acquire the lock because WE might be
Packit Service df60bb
		** that other thread that was too late to create the data and the first thread
Packit Service df60bb
		** in will have created it.
Packit Service df60bb
		*/
Packit Service df60bb
		NXLock(gLibLock);
Packit Service df60bb
Packit Service df60bb
		if (!(app_data = (libdata_t *) get_app_data(id))) {
Packit Service df60bb
			app_data = (libdata_t *) malloc(sizeof(libdata_t));
Packit Service df60bb
Packit Service df60bb
			if (app_data) {
Packit Service df60bb
				memset(app_data, 0, sizeof(libdata_t));
Packit Service df60bb
Packit Service df60bb
				app_data->tenbytes = malloc(10);
Packit Service df60bb
				app_data->lock     = NXMutexAlloc(0, 0, &liblock);
Packit Service df60bb
Packit Service df60bb
				if (!app_data->tenbytes || !app_data->lock) {
Packit Service df60bb
					if (app_data->lock)
Packit Service df60bb
						NXMutexFree(app_data->lock);
Packit Service df60bb
Packit Service df60bb
					free(app_data);
Packit Service df60bb
					app_data = (libdata_t *) NULL;
Packit Service df60bb
					err      = ENOMEM;
Packit Service df60bb
				}
Packit Service df60bb
Packit Service df60bb
				if (app_data) {
Packit Service df60bb
					/*
Packit Service df60bb
					** Here we burn in the application data that we were trying to get by calling
Packit Service df60bb
					** get_app_data(). Next time we call the first function, we'll get this data
Packit Service df60bb
					** we're just now setting. We also go on here to establish the per-thread data
Packit Service df60bb
					** for the calling thread, something we'll have to do on each application
Packit Service df60bb
					** thread the first time it calls us.
Packit Service df60bb
					*/
Packit Service df60bb
					err = set_app_data(gLibId, app_data);
Packit Service df60bb
Packit Service df60bb
					if (err) {
Packit Service df60bb
						free(app_data);
Packit Service df60bb
						app_data = (libdata_t *) NULL;
Packit Service df60bb
						err      = ENOMEM;
Packit Service df60bb
					} else {
Packit Service df60bb
						/* create key for thread-specific data... */
Packit Service df60bb
						err = NXKeyCreate(DisposeThreadData, (void *) NULL, &key);
Packit Service df60bb
Packit Service df60bb
						if (err)                /* (no more keys left?) */
Packit Service df60bb
							key = -1;
Packit Service df60bb
Packit Service df60bb
						app_data->perthreadkey = key;
Packit Service df60bb
					}
Packit Service df60bb
				}
Packit Service df60bb
			}
Packit Service df60bb
		}
Packit Service df60bb
Packit Service df60bb
		NXUnlock(gLibLock);
Packit Service df60bb
	}
Packit Service df60bb
Packit Service df60bb
	if (app_data) {
Packit Service df60bb
		key = app_data->perthreadkey;
Packit Service df60bb
Packit Service df60bb
		if (     key != -1                  /* couldn't create a key? no thread data */
Packit Service df60bb
		         && !(err = NXKeyGetValue(key, (void **) &thread_data))
Packit Service df60bb
		         && !thread_data) {
Packit Service df60bb
			/*
Packit Service df60bb
			** Allocate the per-thread data for the calling thread. Regardless of whether
Packit Service df60bb
			** there was already application data or not, this may be the first call by a
Packit Service df60bb
			** a new thread. The fact that we allocation 20 bytes on a pointer is not very
Packit Service df60bb
			** important, this just helps to demonstrate that we can have arbitrarily
Packit Service df60bb
			** complex per-thread data.
Packit Service df60bb
			*/
Packit Service df60bb
			thread_data = (libthreaddata_t *) malloc(sizeof(libthreaddata_t));
Packit Service df60bb
Packit Service df60bb
			if (thread_data) {
Packit Service df60bb
				thread_data->_errno      = 0;
Packit Service df60bb
				thread_data->twentybytes = malloc(20);
Packit Service df60bb
Packit Service df60bb
				if (!thread_data->twentybytes) {
Packit Service df60bb
					free(thread_data);
Packit Service df60bb
					thread_data = (libthreaddata_t *) NULL;
Packit Service df60bb
					err         = ENOMEM;
Packit Service df60bb
				}
Packit Service df60bb
Packit Service df60bb
				if ((err = NXKeySetValue(key, thread_data))) {
Packit Service df60bb
					free(thread_data->twentybytes);
Packit Service df60bb
					free(thread_data);
Packit Service df60bb
					thread_data = (libthreaddata_t *) NULL;
Packit Service df60bb
				}
Packit Service df60bb
			}
Packit Service df60bb
		}
Packit Service df60bb
	}
Packit Service df60bb
Packit Service df60bb
	if (appData)
Packit Service df60bb
		*appData = app_data;
Packit Service df60bb
Packit Service df60bb
	if (threadData)
Packit Service df60bb
		*threadData = thread_data;
Packit Service df60bb
Packit Service df60bb
	return err;
Packit Service df60bb
}
Packit Service df60bb
Packit Service df60bb
int DisposeLibraryData
Packit Service df60bb
(
Packit Service df60bb
    void    *data
Packit Service df60bb
)
Packit Service df60bb
{
Packit Service df60bb
	if (data) {
Packit Service df60bb
		void    *tenbytes = ((libdata_t *) data)->tenbytes;
Packit Service df60bb
Packit Service df60bb
		if (tenbytes)
Packit Service df60bb
			free(tenbytes);
Packit Service df60bb
Packit Service df60bb
		free(data);
Packit Service df60bb
	}
Packit Service df60bb
Packit Service df60bb
	return 0;
Packit Service df60bb
}
Packit Service df60bb
Packit Service df60bb
void DisposeThreadData
Packit Service df60bb
(
Packit Service df60bb
    void    *data
Packit Service df60bb
)
Packit Service df60bb
{
Packit Service df60bb
	if (data) {
Packit Service df60bb
		void    *twentybytes = ((libthreaddata_t *) data)->twentybytes;
Packit Service df60bb
Packit Service df60bb
		if (twentybytes)
Packit Service df60bb
			free(twentybytes);
Packit Service df60bb
Packit Service df60bb
		free(data);
Packit Service df60bb
	}
Packit Service df60bb
}
Packit Service df60bb
Packit Service df60bb