checkdll.c   [plain text]


/*
 * checkdll.c - Winsock DLL/IDN processing status
 */

/*
 * Copyright (c) 2000,2002 Japan Network Information Center.
 * All rights reserved.
 *  
 * By using this file, you agree to the terms and conditions set forth bellow.
 * 
 * 			LICENSE TERMS AND CONDITIONS 
 * 
 * The following License Terms and Conditions apply, unless a different
 * license is obtained from Japan Network Information Center ("JPNIC"),
 * a Japanese association, Kokusai-Kougyou-Kanda Bldg 6F, 2-3-4 Uchi-Kanda,
 * Chiyoda-ku, Tokyo 101-0047, Japan.
 * 
 * 1. Use, Modification and Redistribution (including distribution of any
 *    modified or derived work) in source and/or binary forms is permitted
 *    under this License Terms and Conditions.
 * 
 * 2. Redistribution of source code must retain the copyright notices as they
 *    appear in each source code file, this License Terms and Conditions.
 * 
 * 3. Redistribution in binary form must reproduce the Copyright Notice,
 *    this License Terms and Conditions, in the documentation and/or other
 *    materials provided with the distribution.  For the purposes of binary
 *    distribution the "Copyright Notice" refers to the following language:
 *    "Copyright (c) 2000-2002 Japan Network Information Center.  All rights reserved."
 * 
 * 4. The name of JPNIC may not be used to endorse or promote products
 *    derived from this Software without specific prior written approval of
 *    JPNIC.
 * 
 * 5. Disclaimer/Limitation of Liability: THIS SOFTWARE IS PROVIDED BY JPNIC
 *    "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 JPNIC 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 DAMAGES.
 */

#include <windows.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "wrapcommon.h"

static int winsock_idx;		/* index of winsock_info[] */

static struct winsock_type {
	char *version;		/* winsock version */
	char *name;		/* wrapper DLL name */
	char *original_name;	/* original DLL name */
} winsock_info[] = {
#define IDN_IDX_WS11 0
	{ "1.1", "WSOCK32", "WSOCK32O" },
#define IDN_IDX_WS20 1
	{ "2.0", "WS2_32",  "WS2_32O" },
	{ NULL, NULL, NULL },
};

static HINSTANCE	load_original_dll(void);
static BOOL		check_idn_processing(void);
static BOOL		check_dll(const char *name);

BOOL
idnWinsockVersion(const char *version) {
	int i;
	for (i = 0; winsock_info[i].version != NULL; i++) {
		if (strcmp(winsock_info[i].version, version) == 0) {
			winsock_idx = i;
			idnLogPrintf(idn_log_level_trace,
				     "idnWinsockVersion: version %s\n",
				     version);
			return (TRUE);
		}
	}
	idnLogPrintf(idn_log_level_fatal,
		     "idnWinsockVersion: unknown winsock version %s\n",
		     version);
	return (FALSE);
}

HINSTANCE
idnWinsockHandle(void) {
	static HINSTANCE dll_handle = NULL;
	static int initialized = 0;

	if (!initialized) {
		/* Get the handle of the original winsock DLL */
		idnLogPrintf(idn_log_level_trace,
			     "idnWinsockHandle: loading original DLL..\n");
		dll_handle = load_original_dll();
	}
	initialized = 1;
	return (dll_handle);
}

idn_resconf_t
idnGetContext(void) {
	static int initialized = 0;
	static idn_resconf_t ctx = NULL;

	if (!initialized) {
		/*
		 * Check whether IDN processing should be done
		 * in this wrapper DLL.
		 */
		idnLogPrintf(idn_log_level_trace,
			     "idnGetContext: checking IDN status..\n");
		if (check_idn_processing()) {
			/* Initialize idnkit */
			ctx = idnConvInit();
			idnLogPrintf(idn_log_level_info,
				     "Processing context: %08x\n", ctx);
		} else {
			idnLogPrintf(idn_log_level_info,
				     "NOT process IDN here\n");
			ctx = NULL;
		}
		initialized = 1;
	}

	return (ctx);
}

static HINSTANCE
load_original_dll(void) {
	/*
	 * Load Original DLL
	 */
	char dllpath[MAX_PATH];
	const char *dll_name = winsock_info[winsock_idx].original_name;
	HINSTANCE handle;

	/*
	 * Get idn wrapper's install directory, where the copies of
	 * the original winsock DLLs are saved.
	 */
	dllpath[0] = '\0';
	if (idnGetInstallDir(dllpath, sizeof(dllpath)) != TRUE) {
		idnLogPrintf(idn_log_level_fatal,
			     "idnWinsockHandle: cannot find idn wrapper's "
			     "install directory\n");
		abort();
		return (NULL);	/* for lint */
	}
	/* Strip the trailing backslash. */
	if (dllpath[0] != '\0' &&
	    dllpath[strlen(dllpath) - 1] == '\\') {
		dllpath[strlen(dllpath) - 1] = '\0';
	}
	/* Is the pathname is insanely long? */
	if (strlen(dllpath) + strlen(dll_name) + 1 + 4 >= sizeof(dllpath)) {
		idnLogPrintf(idn_log_level_fatal,
			     "idnWinsockHandle: idn wrapper's install path is "
			     "too long to be true\n");
		abort();
		return (NULL);	/* for lint */
	}
	/* Append the DLL name to form a full pathname of the DLL. */
	strcat(dllpath, "\\");
	strcat(dllpath, dll_name);
	strcat(dllpath, ".DLL");

	idnLogPrintf(idn_log_level_trace,
		     "idnWinsockHandle: loading original winsock DLL (%s)\n",
		     dllpath);
	if ((handle = LoadLibrary(dllpath)) == NULL) {
		idnLogPrintf(idn_log_level_fatal,
			     "idnWinsockHandle: no DLL %-.100s\n", dllpath);
		abort();
		return (NULL);	/* font lint */
	}
	return (handle);
}

static BOOL
check_idn_processing(void) {
	int where = idnEncodeWhere();
	BOOL here = FALSE;

	idnLogPrintf(idn_log_level_trace,
		     "idnGetContext: Winsock%s, where=%d\n",
		     winsock_info[winsock_idx].version, where);

	switch (winsock_idx) {
	case IDN_IDX_WS11:
		switch (where) {
		case IDN_ENCODE_ALWAYS:
		case IDN_ENCODE_ONLY11:
			return (TRUE);
		case IDN_ENCODE_CHECK:
			if (!check_dll(winsock_info[winsock_idx].name)) {
				return (TRUE);
			}
			break;
		}
		break;
	case IDN_IDX_WS20:
		switch (where) {
		case IDN_ENCODE_ALWAYS:
		case IDN_ENCODE_ONLY20:
		case IDN_ENCODE_CHECK:
			return (TRUE);
			break;
		}
		break;
	}
	return (FALSE);
}

static BOOL
check_dll(const char *name) {
	HINSTANCE hdll = NULL;

#if 1
	hdll = LoadLibrary(name);
#else
	/*
	 * Just check the existence of the named DLL, without taking
	 * the trouble of calling DllMain.
	 */
	hdll = LoadLibraryEx(name, NULL, LOAD_LIBRARY_AS_DATAFILE);
#endif
	if (hdll == NULL) {
		idnLogPrintf(idn_log_level_trace,
			     "idnGetContext: DLL %s does not exist\n");
		return (FALSE);
	} else {
		idnLogPrintf(idn_log_level_trace,
			     "idnGetContext: DLL %s exists\n");
		FreeLibrary(hdll);
		return (TRUE);
	}
}