[Author Prev][Author Next][Thread Prev][Thread Next][Author Index][Thread Index]

[or-cvs] r6917: Initial import for Mike Chiussi's bsockets library. (/ bsockets bsockets/trunk)



Author: nickm
Date: 2006-07-26 23:47:11 -0400 (Wed, 26 Jul 2006)
New Revision: 6917

Added:
   bsockets/
   bsockets/branches/
   bsockets/tags/
   bsockets/trunk/
   bsockets/trunk/LICENSE
   bsockets/trunk/Makefile
   bsockets/trunk/bsocket.h
   bsockets/trunk/callback.c
   bsockets/trunk/callback.h
   bsockets/trunk/event.c
   bsockets/trunk/event.h
   bsockets/trunk/io.c
   bsockets/trunk/io.h
   bsockets/trunk/list.c
   bsockets/trunk/list.h
   bsockets/trunk/misc.c
   bsockets/trunk/misc.h
   bsockets/trunk/notes
   bsockets/trunk/select.c
   bsockets/trunk/select.h
   bsockets/trunk/socket.c
   bsockets/trunk/socket.h
   bsockets/trunk/sync.c
   bsockets/trunk/sync.h
   bsockets/trunk/test.c
   bsockets/trunk/test.h
   bsockets/trunk/todo
   bsockets/trunk/unix.c
   bsockets/trunk/unix.h
   bsockets/trunk/wait.c
   bsockets/trunk/wait.h
   svn-commit.tmp
Log:
Initial import for Mike Chiussi's bsockets library.


Added: bsockets/trunk/LICENSE
===================================================================
--- bsockets/trunk/LICENSE	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/LICENSE	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,33 @@
+==============================================================================
+BSOCK is distributed under this license
+
+Copyright (c) 2006 Mike Chiussi
+
+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.
+
+    * Redistributions in binary form must reproduce the above
+copyright notice, this list of conditions and the following disclaimer
+in the documentation and/or other materials provided with the
+distribution.
+
+    * Neither the names of the copyright owners nor the names of its
+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.
+==============================================================================
\ No newline at end of file


Property changes on: bsockets/trunk/LICENSE
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/Makefile
===================================================================
--- bsockets/trunk/Makefile	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/Makefile	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,23 @@
+test_HEADERS= test.h
+bsock_HEADERS = list.h test.h socket.h bsocket.h misc.h
+
+sock_OBJS = list.o socket.o unix.o event.o sync.o select.o wait.o misc.o io.o callback.o
+test_OBJS = test.o ${sock_OBJS}
+
+BIN_SUFFIX = .exe
+
+default: test libbsock.a 
+
+clean:
+	rm -f *.exe *.a *.o
+	
+libbsock.a: ${sock_OBJS}
+	ar -r $@ ${sock_OBJS} 
+
+test: ${test_OBJS} 
+	gcc -Wall -o test${BIN_SUFFIX} ${test_OBJS} -lws2_32
+
+
+%.o: %.c %.h ${test_HEADERS} ${bsock_HEADERS} 
+	gcc -g -Wall -c $<	
+	 


Property changes on: bsockets/trunk/Makefile
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/bsocket.h
===================================================================
--- bsockets/trunk/bsocket.h	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/bsocket.h	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,97 @@
+#ifndef _BSOCKET_H_
+#define _BSOCKET_H_
+
+#define USE_WIN32
+
+#ifdef USE_WIN32
+
+#define MAX_BSOCKETS 2048
+
+typedef struct _BFD_SET {
+	int hash[MAX_BSOCKETS];
+	int list[MAX_BSOCKETS];
+	int count;
+	int i;
+} bfd_set;
+
+
+#define BFD_COPY(X,Y)	*(X) = *(Y);
+
+#define	BFD_ZERO(X)		(X)->count = 0;\
+
+#define BFD_ISSET(F,X)	(((X)->hash[F] < (X)->count)\
+							&&\
+						((X)->list[(X)->hash[F]] == F))\
+
+#define BFD_SET(F,X)		if (!BFD_ISSET(F,X)) {\
+							(X)->hash[F] = (X)->count;\
+							(X)->list[(X)->count] = F;\
+							(X)->count++;\
+						}\
+
+#define BFD_CLR(F,X)	if ( BFD_ISSET(F,X) ) {\
+							(X)->i=(X)->hash[F];\
+							for (; (X)->i<(X)->count-1; (X)->i++) {\
+								(X)->list[(X)->i] = (X)->list[(X)->i+1];\
+								((X)->hash[(X)->list[(X)->i]])--;\
+							}\
+							(X)->count--;\
+						}\
+
+struct socket_env;
+struct socket_env *__GLOBAL_BSOCKET_ENV_;
+
+int winsock_start();
+
+int socket_win32(int, int, int, struct socket_env *);
+int close_win32(int, struct socket_env *);
+int connect_win32(int, struct sockaddr*,size_t,struct socket_env*);
+int socket_init_win32(struct socket_env*);
+int socket_cleanup_win32(struct socket_env*);
+int fcntl_win32(int,int,long, struct socket_env*);
+int select_win32(int, bfd_set*, bfd_set*, bfd_set*, struct timeval*, struct socket_env*);
+int getsockopt_win32(int, int, int, void*, int *, struct socket_env*);
+int send_win32(int, void*, size_t, int, struct socket_env*);
+int recv_win32(int, void*, size_t, int, struct socket_env*);
+
+#define bsocket_init(X)			socket_init_win32(X)
+#define bsocket_shutdown(X)		socket_cleanup_win32(X)
+
+#define bsocket(A,B,C)			socket_win32(A,B,C,__GLOBAL_BSOCKET_ENV_)
+#define bconnect(A,B,C)			connect_win32(A,B,C,__GLOBAL_BSOCKET_ENV_)
+#define bclose(X)				close_win32(X,__GLOBAL_BSOCKET_ENV_)
+
+#define bfcntl(X,Y,Z)			fcntl_win32(X,Y,Z,__GLOBAL_BSOCKET_ENV_)
+
+#define bselect(W,X,Y,Z,A)		select_win32(W,X,Y,Z,A,__GLOBAL_BSOCKET_ENV_)
+
+#define bgetsockopt(A,B,C,D,E)	getsockopt_win32(A,B,C,D,E,__GLOBAL_BSOCKET_ENV_)
+
+#define bsend(A,B,C,D)			send_win32(A,B,C,D,__GLOBAL_BSOCKET_ENV_)
+#define brecv(A,B,C,D)			recv_win32(A,B,C,D,__GLOBAL_BSOCKET_ENV_);
+
+#define F_SETFL					1
+
+#define	O_NONBLOCK				1
+#define O_BLOCK					2
+
+/**************/
+/*user options*/
+/**************/
+
+/*these should be throttled carefully, as too many open half-open sockets
+	or listening sockets wreak havoc on non-server kernels*/
+
+#define MAX_SIMUL_CONNECT 		9
+#define MAX_SIMUL_LISTEN  		5
+
+#if (MAX_SIMUL_CONNECT + MAX_SIMUL_LISTEN + 2 > WSA_MAXIMUM_WAIT_EVENTS )
+#error "Reduce MAX_SIMUL_CONNECT or MAX_SIMUL_LISTEN"
+#endif
+
+
+#else
+#endif
+
+#endif
+


Property changes on: bsockets/trunk/bsocket.h
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/callback.c
===================================================================
--- bsockets/trunk/callback.c	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/callback.c	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,111 @@
+#include <windows.h>
+
+#include "misc.h"
+#include "socket.h"
+#include "event.h"
+#include "wait.h"
+#include "select.h"
+#include "io.h"
+#include "list.h"
+
+void CALLBACK callback_write(
+	 IN DWORD err,
+	 IN DWORD len,
+	 IN WSAOVERLAPPED *wo,
+	 IN DWORD flags) {
+
+	//todo -- activate write_wait_list
+	//todo -- if write is incomplete, reiterate
+	//todo -- if error, raise exception
+
+}
+
+void CALLBACK callback_read(
+	IN DWORD err,
+	IN DWORD len,
+	IN WSAOVERLAPPED *wo,
+	IN DWORD flags) {
+
+	struct bsocket *b;
+	struct _msg *msg;
+
+	msg = (struct _msg*) wo->hEvent;
+
+	if (err == 0) {
+
+		ASSERT(len >= 0);
+
+
+		ASSERT(len <= MSG_SIZE);
+
+		b = bsocket_get(msg->fd,AS_READ,msg->env);
+
+		ASSERT(b != NULL);
+		ASSERT(b->fd == msg->fd);
+
+		if ( list_enqueue(msg,b->in_q ) == 0) {
+
+			msg->len = len;
+			printf("%d\n",msg->len);
+			fflush(stdout);
+
+			post_read(b,msg->env);
+			socket_raise(b->fd,IS_READABLE,TRUE,msg->env);
+
+		} else {
+
+			//todo -- dealloc msg?
+			socket_exception(b->fd,errno,msg->env);
+		}
+
+		bsocket_release(msg->fd,AS_READ,msg->env);
+
+	} else {
+
+		socket_exception(msg->fd,unixify_wsaerr(err),msg->env);
+	}
+
+}
+
+//caller must guarantee atomicity
+void make_connected(struct bsocket *b, struct bsocket *partner, struct socket_env *env) {
+
+	b->connected = TRUE;
+	b->partner = partner;
+
+	//todo -- start reading
+	bsocket_raise(b,IS_READABLE,FALSE,env);
+	bsocket_raise(b,IS_WRITABLE,TRUE,env);
+
+	post_read(b,env);
+
+}
+
+void complete_connect(int fd, int err, struct socket_env *env) {
+
+	struct bsocket **l;
+
+	l = bsocket_list_get(env);
+
+	if (l != NULL) {
+
+		if (l[fd] != NULL) {
+
+			l[fd]->connecting = FALSE;
+
+			if (err) {
+
+				bsocket_exception(l[fd],err,env);
+
+			} else {
+
+				make_connected(l[fd],NULL,env);
+		}
+
+		}
+
+		bsocket_list_release(env);
+
+	}
+
+}


Property changes on: bsockets/trunk/callback.c
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/callback.h
===================================================================
--- bsockets/trunk/callback.h	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/callback.h	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,10 @@
+#ifndef _CALLBACK_H_
+#define _CALLBACK_H_
+
+
+void CALLBACK callback_write(DWORD, DWORD, OVERLAPPED*, DWORD);
+void CALLBACK callback_read (DWORD, DWORD, OVERLAPPED*, DWORD);
+
+void complete_connect(int, int, struct socket_env *);
+#endif
+


Property changes on: bsockets/trunk/callback.h
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/event.c
===================================================================
--- bsockets/trunk/event.c	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/event.c	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,648 @@
+#include <windows.h>
+#include <stdio.h>
+
+#include "list.h"
+#include "event.h"
+#include "socket.h"
+#include "misc.h"
+#include "select.h"
+#include "callback.h"
+#include "io.h"
+
+struct _connect_data *connect_data_new(struct sockaddr *name, size_t len) {
+
+	struct _connect_data *out;
+
+	out = (struct _connect_data *)
+		malloc(sizeof(struct _connect_data));
+
+	if (out != NULL) {
+
+		out->name = malloc(len);
+
+		if (out->name != NULL) {
+
+			memcpy(out->name,name,len);
+			out->len  = len;
+
+		} else {
+
+			errno = ENOMEM;
+			free(out);
+			out = NULL;
+
+		}
+
+	} else {
+		errno = ENOMEM;
+	}
+
+	return out;
+
+}
+
+void connect_data_free(struct _connect_data *d) {
+	free(d->name);
+	free(d);
+}
+
+void event_free(struct _event *e) {
+
+	ASSERT(e != NULL);
+
+	switch (e->type) {
+
+		case EV_CONNECT:
+
+			if (e->data != NULL) {
+				connect_data_free(e->data);
+			}
+
+		break;
+
+		case EV_CLOSE:
+		case EV_SHUTDOWN:
+		case EV_PING:
+		case EV_EXCEPTION:
+			//nothing to do
+		break;
+
+		case EV_WRITE:
+			//nothing to do
+		break;
+
+		default:
+			ASSERT(FALSE);
+		break;
+
+	}
+
+	free(e);
+
+}
+
+void event_respond(struct _event *e, int ret, int err) {
+
+	int out;
+
+
+	if (ret) {
+		out = ret;
+	} else {
+		out = 0;
+	}
+
+	if (e->ret != NULL) {
+
+		*e->ret = out;
+
+	}
+
+	if (e->err != NULL) {
+		*e->err = err;
+	}
+
+
+	if (e->sig != NULL) {
+
+		ASSERT(SetEvent(e->sig));
+	}
+
+
+	event_free(e);
+
+}
+
+//if a bsocket is passed, it is the users responsibility to ensure it remains locked
+//during the event processing
+int event_post(struct bsocket *b, int type, void *data, int *ret, int *err, int wait, struct socket_env *env) {
+
+	HANDLE w;
+	struct _event *e;
+
+	int out;
+
+	out = 0;
+	e = (struct _event*)
+		malloc(sizeof(struct _event));
+
+	CHECK(e != NULL,ENOMEM);
+
+	if (wait) {
+		w = CreateEvent(NULL,TRUE,FALSE,NULL);
+		CHECK_(w != NULL);
+	} else {
+		w = NULL;
+	}
+
+	if (b != NULL) {
+		e->s = b->s;
+		e->fd = b->fd;
+	}
+
+	e->type = type;
+	e->data = data;
+	e->ret = ret;
+	e->sig = w;
+	e->err = err;
+
+
+	MUTEX_ACQUIRE(env->post_m);
+
+	if (list_enqueue(e,env->event_q) == NULL) {
+		out = -1;
+	}
+
+	ASSERT(WSASetEvent(env->post_e));
+
+	MUTEX_RELEASE(env->post_m);
+
+	if (w != NULL) {
+		ASSERT( WaitForSingleObject(w,INFINITE) == WAIT_OBJECT_0);
+	}
+
+	fail:
+
+
+	if (out == -1) {
+		e->type = EV_PING;
+		event_free(e);
+	}
+
+
+	return out;
+
+}
+
+struct _event *event_copy(struct _event *e) {
+
+	struct _event *out;
+
+	out = (struct _event*) malloc(sizeof(struct _event));
+
+	if (out != NULL ) {
+
+		out->s = e->s;
+		out->fd = e->fd;
+		out->type = e->type;
+		out->data = e->data;
+
+		out->ret = NULL;
+		out->sig = NULL;
+		out->err = NULL;
+
+	} else {
+
+		errno = ENOMEM;
+
+	}
+
+	return out;
+
+}
+
+
+void event_ping(struct socket_env *env) {
+
+	ASSERT( event_post(NULL,EV_PING,NULL,NULL,NULL,TRUE,env) == 0);
+
+}
+
+int post_connect(struct bsocket *b, struct sockaddr *name, size_t len, struct socket_env *env) {
+
+	struct _connect_data *c;
+
+	int out;
+	int r;
+	int ret;
+
+	out = 0;
+	c = connect_data_new(name,len);
+
+	CHECK(c != NULL,0);
+
+	r = event_post(b, EV_CONNECT,c,&ret,NULL,TRUE,env);
+
+	CHECK(r == 0,0);
+
+	if (ret) {
+		errno = ret;
+		out = -1;
+	} else {
+		out = 0;
+	}
+
+	fail:
+	return out;
+}
+
+int post_write(struct bsocket *b, int *ret, int *err, struct socket_env *env) {
+
+	ASSERT(b->write_buf != NULL);
+
+	b->write_wb.buf = b->write_buf;
+	b->write_wb.len = b->write_len;
+
+	return  event_post(b,EV_WRITE,&b->write_wb,ret,err,TRUE,env);
+
+
+}
+
+int post_exception(struct bsocket *b, int err, struct socket_env *env) {
+
+	return event_post(b,EV_EXCEPTION,(void*) err,NULL,NULL,FALSE,env);
+
+}
+
+int post_close(struct bsocket *b, struct socket_env *env) {
+
+	return post_exception(b,ECLOSED,env);
+
+}
+
+int post_read(struct bsocket *b, struct socket_env *env) {
+
+	return event_post(b,EV_READ,NULL,NULL,NULL,FALSE,env);
+
+}
+
+struct _event *event_next(struct socket_env *env) {
+
+	struct _event *e;
+
+	MUTEX_ACQUIRE(env->post_m);
+
+	if (list_dequeue(&e,env->event_q)) {
+
+		WSAResetEvent(env->post_e);
+		e = NULL;
+		errno = EEMPTY;
+	}
+
+	MUTEX_RELEASE(env->post_m);
+
+	return e;
+
+}
+
+//todo -- make this less hideous
+//todo -- is it possible for there to be events related to a socket after closure?
+
+#define E_NOTIFY	(0+WAIT_OBJECT_0)
+#define E_CONNECT	(1+WAIT_OBJECT_0)
+#define E_ACCEPT	(2+WAIT_OBJECT_0)
+
+#define E_INSERT(S,FD,TYPE) 	for (_j=1; _j<WSA_MAXIMUM_WAIT_EVENTS; ) {\
+									if (ev[_j].fd == -1) {\
+										ASSERT( WSAEventSelect(S,we[_j],TYPE) ==0 );\
+										ev[_j].fd	= FD;\
+										ev[_j].s 	= S;\
+										ev[_j].type	= TYPE;\
+										switch(TYPE) {\
+											case FD_CONNECT:\
+												connect_lookup[(int)FD] = _j;\
+											break;\
+											case FD_ACCEPT:\
+												listen_lookup[(int)FD] = _j;\
+											break;\
+											default:\
+												ASSERT(FALSE);\
+											break;\
+										}\
+										_j = WSA_MAXIMUM_WAIT_EVENTS+2;\
+									} else {\
+										_j++;\
+									}\
+								}\
+								ASSERT(_j == (WSA_MAXIMUM_WAIT_EVENTS+2));\
+
+#define E_DELETE(X)				ASSERT(X>0); ASSERT(X<WSA_MAXIMUM_WAIT_EVENTS);\
+								ASSERT( WSAEventSelect(ev[X].s,we[X],0) == 0);\
+								connect_lookup[(int) ev[X].fd] = -1;\
+								listen_lookup[(int) ev[X].fd]  = -1;\
+								ev[X].fd = -1;\
+
+struct __ev {
+	SOCKET s;
+	int fd;
+	int type;
+};
+
+void event_manager(struct socket_env *env) {
+
+	SOCKET connect_lookup[MAX_BSOCKETS];
+	SOCKET listen_lookup[MAX_BSOCKETS];
+
+	WSAEVENT we[WSA_MAXIMUM_WAIT_EVENTS];
+	struct __ev ev[WSA_MAXIMUM_WAIT_EVENTS];
+
+	WSANETWORKEVENTS ne;
+
+	WSAOVERLAPPED *wo;
+
+	struct _connect_data *c;
+	struct _event *e2;
+	struct _event *e;
+
+	struct _list *connect_q;
+
+	int i, _j;
+	int err;
+	int out;
+	int resume;
+	int r,z;
+	int type;
+
+	int x;
+	int connect_count;
+
+	connect_q = NULL;
+
+	we[0] 	= env->post_e;
+
+	resume = TRUE;
+	connect_count = 0;
+
+
+	connect_q = list_new();
+
+	CHECK(connect_q != NULL, 0);
+
+	for (i=1; i<WSA_MAXIMUM_WAIT_EVENTS; i++) {
+
+		we[i] = WSACreateEvent();
+		CHECK_(we[i] != WSA_INVALID_EVENT);
+
+		ev[i].fd = -1;
+
+	}
+
+	x = 0;
+
+	for (i=0; i<MAX_BSOCKETS; i++) {
+		connect_lookup[i] = (SOCKET) -1;
+		listen_lookup[i]  = (SOCKET) -1;
+	}
+
+	while (resume) {
+
+		r = WSAWaitForMultipleEvents(WSA_MAXIMUM_WAIT_EVENTS,we,FALSE,INFINITE,TRUE);
+
+		ASSERT(connect_count <= MAX_SIMUL_CONNECT);
+
+		switch (r) {
+
+			case E_NOTIFY:
+
+				e = event_next(env);
+
+				if (e == NULL) {
+					ASSERT(errno == EEMPTY);
+					break;
+				}
+
+				switch (e->type) {
+
+					case EV_PING:
+						event_respond(e,0,0);
+					break;
+
+					case EV_EXCEPTION:
+
+						z = 0;
+
+						if ((i = connect_lookup[(SOCKET) ev->fd]) != -1) {
+							E_DELETE(i);
+							z++;
+						}
+
+						if ((i = listen_lookup[(SOCKET) ev->fd]) != -1) {
+							E_DELETE(i);
+							z++;
+						}
+
+						ASSERT(z <= 1);
+
+						//todo cancel any outstanding i/o (do we need to?)
+						ASSERT(closesocket(e->s) == 0);
+
+						event_respond(e,0,0);
+
+					break;
+
+					case EV_SHUTDOWN:
+						resume = FALSE;
+						event_respond(e,0,0);
+					break;
+
+					case EV_CONNECT:
+
+						//schedule connection request
+						if ( (c = e->data) != NULL) {
+
+							e2 = event_copy(e);
+							e->data = NULL;
+
+							if (e2 != NULL) {
+
+								if (list_enqueue(e2,connect_q) == NULL) {
+
+									event_respond(e,-1,errno);
+
+								} else {
+
+									event_respond(e,0,0);
+
+								}
+
+							} else {
+
+								event_respond(e,-1,errno);
+							}
+
+
+						} else {
+
+							event_respond(e,-1,errno);
+
+						}
+
+						//process as many as we can right now
+						while ( connect_count < MAX_SIMUL_CONNECT
+								&&
+								list_dequeue(&e,connect_q) == 0
+						) {
+							c = e->data;
+
+							z = connect(e->s,c->name,c->len);
+
+							if (z == 0) {
+
+								//todo -- start reading
+								socket_raise(e->fd,IS_WRITABLE,TRUE,env);
+
+							} else {
+
+								if (WSAGetLastError() == WSAEWOULDBLOCK ) {
+
+									E_INSERT(e->s,e->fd,FD_CONNECT);
+
+									connect_count++;
+
+								} else {
+
+								//todo -- is there a memory leak here?
+									complete_connect(
+										e->fd,
+										unixify_wsaerr(WSAGetLastError()),
+										env	);
+
+
+								}
+
+							}
+
+						}
+
+					break;
+
+					case EV_READ:
+
+						invoke_read(e->s,e->fd,env);
+
+					break;
+
+					case EV_WRITE:
+
+						wo = (WSAOVERLAPPED*)
+							malloc(sizeof(WSAOVERLAPPED));
+
+
+						r = WSASend(
+								e->s,
+								(WSABUF*) (e->data),
+								1,
+								(DWORD*) &z,
+								0,
+								wo,
+								callback_write);
+
+						if (r == 0) {
+
+							event_respond(e,z,0);
+
+						} else {
+
+							switch(err = WSAGetLastError()) {
+
+								case WSA_IO_PENDING:
+									event_respond(e,z,0);
+								break;
+
+								default:
+									event_respond(e,-1,unixify_wsaerr(err));
+								break;
+							}
+
+						}
+
+					break;
+
+					case EV_LISTEN:
+						ASSERT(FALSE);
+					break;
+
+
+					default:
+						ASSERT(FALSE);
+					break;
+
+				}
+
+			break;
+
+			default:
+
+				i   = r - WSA_WAIT_EVENT_0;
+
+				ASSERT(i < WSA_MAXIMUM_WAIT_EVENTS);
+
+				z   = WSAEnumNetworkEvents(ev[i].s,we[i],&ne);
+
+				if (z != SOCKET_ERROR) {
+
+					ASSERT( ne.lNetworkEvents & ev[i].type);
+					//make sure no other events are getting set
+					ASSERT( (ne.lNetworkEvents & ~(ev[i].type)) == 0);
+
+					switch (ev[i].type) {
+
+						case FD_CONNECT:
+							err 	= ne.iErrorCode[FD_CONNECT_BIT];
+							type	= IS_WRITABLE;
+						break;
+
+						case FD_ACCEPT:
+							err		= ne.iErrorCode[FD_ACCEPT_BIT];
+							type 	= IS_READABLE;
+						break;
+
+						default:
+							ASSERT(FALSE);
+						break;
+					}
+
+					if (err) {
+
+						complete_connect(ev[i].fd,unixify_wsaerr(err),env);
+
+					} else {
+
+						complete_connect(ev[i].fd,0,env);
+					}
+
+				} else {
+
+
+					complete_connect(ev[i].fd,unixify_wsaerr(err),env);
+
+				}
+
+				if (ev[i].type == FD_CONNECT) {
+
+					//todo -- figure out a way to test the element is being deleted
+
+					E_DELETE(i);
+
+					//if (connect_count >= MAX_SIMUL_CONNECT) {
+					connect_count--;
+
+					ASSERT(event_post(NULL,EV_CONNECT,NULL,NULL,NULL,FALSE,env) == 0);
+
+				}
+
+			break;
+
+			case WSA_WAIT_FAILED:
+			case WSA_WAIT_TIMEOUT:
+				ASSERT(FALSE);
+			break;
+
+			case WAIT_IO_COMPLETION:
+			break;
+
+		}
+
+
+	}
+
+	fail:
+
+	if (connect_q != NULL)
+		list_free(connect_q);
+
+
+}
+
+
+


Property changes on: bsockets/trunk/event.c
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/event.h
===================================================================
--- bsockets/trunk/event.h	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/event.h	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,53 @@
+#ifndef _EVENT_H_
+#define _EVENT_H_
+
+#include "socket.h"
+
+#define EV_PING			(0)
+#define	EV_SOCK			(1)
+#define EV_CLOSE		(2)
+#define EV_CONNECT		(3)
+#define EV_LISTEN		(4)
+#define EV_WRITE		(5)
+#define EV_READ			(6)
+
+#define EV_EXCEPTION	(50)
+
+#define EV_SHUTDOWN		(100)
+
+struct _connect_data {
+
+	void 	*name;
+	size_t	len;
+
+};
+
+struct _event {
+
+	HANDLE sig;
+	SOCKET s;
+
+	void *data;
+	int *ret;
+	int *err;
+
+	int type;
+	int fd;
+
+};
+
+struct _connect_data *connect_data_new(struct sockaddr*,size_t);
+
+void event_ping(struct socket_env *);
+int event_connect(struct bsocket *, struct sockaddr*, size_t, struct socket_env*);
+
+int event_post(struct bsocket *, int, void*, int*, int*, int, struct socket_env*);
+void event_manager(struct socket_env *);
+
+int post_connect(struct bsocket *, struct sockaddr *, size_t len, struct socket_env *env);
+int post_exception(struct bsocket *, int, struct socket_env*);
+int post_read(struct bsocket *, struct socket_env *);
+int post_write(struct bsocket *, int*, int*, struct socket_env*);
+int post_close(struct bsocket *, struct socket_env*);
+
+#endif


Property changes on: bsockets/trunk/event.h
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/io.c
===================================================================
--- bsockets/trunk/io.c	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/io.c	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,295 @@
+#include <windows.h>
+
+#include "bsocket.h"
+#include "callback.h"
+#include "select.h"
+#include "socket.h"
+#include "misc.h"
+#include "event.h"
+#include "list.h"
+#include "io.h"
+
+struct _msg *msg_new() {
+
+	struct _msg *out;
+
+	out = (struct _msg*)
+		malloc(sizeof(struct _msg));
+
+	if (out != NULL) {
+
+		out->data = out->o_data;
+
+		out->buf_wb.buf = out->o_data;
+		out->buf_wb.len = MSG_SIZE;
+
+
+	} else {
+
+		errno = ENOMEM;
+
+	}
+
+	return out;
+
+}
+
+void msg_free(struct _msg *msg) {
+
+	free(msg);
+
+}
+
+
+void invoke_read(SOCKET s, int fd, struct socket_env *env) {
+
+	WSAOVERLAPPED *wo;
+
+	struct _msg *msg;
+
+	int err;
+
+	int out;
+	int r,z;
+	int flags;
+
+	int resume;
+
+	wo = NULL;
+
+	msg = msg_new();
+
+	CHECK(msg != NULL,0);
+
+	wo = (WSAOVERLAPPED*)
+		malloc(sizeof(WSAOVERLAPPED));
+
+	CHECK(wo != NULL,ENOMEM);
+
+	msg->fd = fd;
+	msg->s  = s;
+	msg->env = env;
+
+	wo->hEvent = msg;
+
+	resume = TRUE;
+
+	out = 0;
+
+	flags = 0;
+
+	while (resume) {
+
+		resume = FALSE;
+
+		z = 0;
+
+		r = WSARecv(
+				s,
+				&msg->buf_wb,
+				1,
+				(DWORD*) &z,
+				(DWORD*) &flags,
+				wo,
+				callback_read
+		);
+
+		if (r != SOCKET_ERROR) {
+
+			if (z == 0) {
+
+				out = -1;
+				err = 0;
+				resume = FALSE;
+
+			} else {
+				resume = TRUE;
+
+			}
+
+
+		} else {
+
+			err = WSAGetLastError();
+
+			CHECK(err == WSA_IO_PENDING,err);
+
+		}
+
+	}
+
+	fail:
+
+	if (out == -1) {
+
+		socket_exception(fd,unixify_wsaerr(err), env);
+
+		if (wo != NULL) {
+			free(wo);
+		}
+
+	}
+
+}
+
+//put EOF in input stream
+//atomicity must have write permissions for socket, list ownership not needed (todo ?)
+void bsocket_eof(struct bsocket *b) {
+
+	//NULL data indicates EOF
+	list_enqueue(NULL,b->in_q);
+
+}
+
+//todo -- check is connected
+int recv_win32(int fd, void *buf, size_t len, int flags, struct socket_env *env) {
+
+	struct bsocket *b;
+
+	struct _msg *msg;
+
+	int out;
+	int r;
+
+	int copy_len;
+
+	int space_left;
+	int data_read;
+
+	void *pos;
+
+	CHECK(len > 0, EINVAL);
+
+	b = bsocket_get(fd,AS_READ,env);
+	CHECK(b != NULL,0);
+
+	if ( list_is_empty(b->in_q)) {
+
+		CHECK(b->blocking,EAGAIN);
+
+		bsocket_release(fd,AS_READ,env);
+
+		r = wait_until(b,IS_READABLE,env);
+
+		b = bsocket_get(fd,AS_READ,env);
+		CHECK(b != NULL,0);
+	}
+
+	space_left = len;
+	pos = buf;
+	data_read = 0;
+
+	//todo -- if next message is EOF, return everything up until it
+
+	while (space_left) {
+
+		r = list_queuepeek(&msg,b->in_q);
+		ASSERT(r == 0);
+
+		if (msg != NULL) {
+
+			copy_len = min(space_left,msg->len);
+			ASSERT(copy_len > 0);
+
+			memcpy(pos,msg->data,copy_len);
+
+			msg->len -= copy_len;
+
+			ASSERT(msg->len >= 0);
+
+			if (msg->len == 0 ) {
+
+				msg_free(msg);
+				ASSERT(list_dequeue(NULL,b->in_q) == 0);
+
+			} else {
+				msg->data += copy_len;
+
+			}
+
+			pos 		+= copy_len;
+			space_left	-= copy_len;
+			data_read	+= copy_len;
+
+
+		} else {
+
+			space_left = 0;
+
+		}
+
+
+
+
+	}
+
+	out = data_read;
+
+	fail:
+
+	if (b != NULL) {
+		bsocket_release(fd,AS_READ,env);
+	}
+
+	return out;
+}
+
+int send_socketpair_win32() {
+
+	return -1;
+
+}
+
+int send_win32 (int fd, void *buf, size_t len, int flags, struct socket_env *env) {
+
+	struct bsocket *b;
+
+	int err;
+	int r;
+	int z;
+	int out;
+
+	b = bsocket_get(fd,AS_WRITE,env);
+	CHECK(b != NULL,0);
+
+	//todo -- if socketpair, send_socketpair_win32
+
+	//todo -- is this good enough to guarantee the socket is writable?
+	if (b->blocking == FALSE) {
+		CHECK(b->write_buf == NULL,EAGAIN);
+	}
+
+	r = wait_until(b,IS_WRITABLE,env);
+
+	//todo
+	//bsocket_raise(b,IS_WRITABLE,0,env);
+
+	CHECK(r == 0,0);
+
+	b->write_buf = malloc(len);
+	CHECK(b != NULL,ENOMEM);
+	memcpy(b->write_buf,buf,len);
+
+	b->write_len = len;
+
+	r = post_write(b,&z,&err,env);
+
+	CHECK(r == 0,0);
+
+	out = z;
+
+	if (out == -1)
+		errno = err;
+
+	fail:
+
+	if ( (out == -1) && (b->write_buf != NULL )) {
+		free(b->write_buf);
+	}
+
+	if (b != NULL) {
+		bsocket_release(fd,AS_WRITE,env);
+	}
+
+	return out;
+}


Property changes on: bsockets/trunk/io.c
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/io.h
===================================================================
--- bsockets/trunk/io.h	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/io.h	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,27 @@
+#ifndef _IO_H_
+#define _IO_H_
+
+//keep in increments of kernel page size
+#define MSG_SIZE	4096
+
+
+struct _msg {
+
+	SOCKET s;
+	WSABUF buf_wb;
+
+	struct socket_env *env;
+
+	char o_data[MSG_SIZE];
+
+	int len;
+	int fd;
+
+	void *data;
+
+};
+
+void bsocket_eof(struct bsocket *);
+void invoke_read(SOCKET s, int fd, struct socket_env *);
+
+#endif


Property changes on: bsockets/trunk/io.h
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/list.c
===================================================================
--- bsockets/trunk/list.c	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/list.c	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,150 @@
+#include <stdlib.h>
+#include <errno.h>
+#include "list.h"
+
+
+int list_is_empty(struct _list *list) {
+
+	return list->head == NULL;
+
+}
+
+void list_data(void *data, struct _list_node *n) {
+		//todo -- this is ugly, why?
+		*((void**) data) = n->data;
+
+}
+
+struct _list_node *list_enqueue(void *data, struct _list *list) {
+
+	struct _list_node *n;
+	struct _monitor *m;
+
+	n = (struct _list_node*) malloc(sizeof(struct _list_node));
+
+	if (n != NULL) {
+
+		m = data;
+
+		n->next = NULL;
+		n->prev = list->tail;
+		n->parent = list;
+
+		n->data = data;
+
+		if (list->tail != NULL) {
+			list->tail->next = n;
+		}
+
+		list->tail = n;
+
+		if (list->head == NULL) {
+			list->head = n;
+		}
+
+	} else {
+		errno = ENOMEM;
+	}
+
+	return n;
+
+}
+
+void list_node_detach(struct _list_node *n) {
+
+	struct _list_node *ne;
+	struct _list_node *pr;
+
+	struct _list *list;
+
+	list = n->parent;
+
+	ne = n->next;
+	pr = n->prev;
+
+	if (ne != NULL) {
+		ne->prev = pr;
+	}
+
+	if (pr != NULL) {
+		pr->next = ne;
+	}
+
+	if (n == list->head) {
+		list->head = ne;
+	}
+
+	if (n == list->tail) {
+		list->tail = pr;
+	}
+
+	list_node_free(n);
+
+}
+
+int list_queuepeek(void *data, struct _list *list) {
+
+	int out;
+
+	if (list->head != NULL) {
+
+		list_data(data,list->head);
+
+		out = 0;
+	} else {
+		out = 1;
+	}
+
+	return out;
+}
+
+int list_dequeue(void *data, struct _list *list) {
+
+	int out;
+
+	if ( (out = list_queuepeek(data,list)) == 0) {
+
+		list_node_detach(list->head);
+
+	}
+
+	return out;
+}
+
+
+void list_node_free(struct _list_node *node) {
+
+	free(node);
+
+}
+
+
+
+struct _list *list_new() {
+
+	struct _list *out;
+
+	out = (struct _list*)
+		malloc(sizeof(struct _list));
+
+	if (out != NULL) {
+
+		out->head = NULL;
+		out->tail = NULL;
+
+	} else {
+		errno = ENOMEM;
+	}
+
+	return(out);
+
+}
+
+void list_free(struct _list *list) {
+
+	int n;
+
+	while (list_dequeue(&n,list) == 0);
+	free(list);
+
+}


Property changes on: bsockets/trunk/list.c
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/list.h
===================================================================
--- bsockets/trunk/list.h	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/list.h	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,41 @@
+#ifndef _LIST_H_
+#define _LIST_H_
+
+
+struct _list_node {
+
+	struct _list_node *next;
+	struct _list_node *prev;
+
+	struct _list *parent;
+
+	void *data;
+
+};
+
+
+struct _list {
+
+	struct _list_node *head;
+	struct _list_node *tail;
+
+	//some optional data to be associated with the list (ie, list id)
+	void *data;
+
+};
+
+struct _list *list_new();
+void list_free(struct _list *);
+
+struct _list_node *list_enqueue(void*, struct _list *);
+int list_dequeue(void*, struct _list *);
+int list_queuepeek(void*, struct _list *);
+int list_is_empty(struct _list *);
+
+void list_node_detach(struct _list_node *);
+
+void list_node_free(struct _list_node *);
+
+
+#endif
+


Property changes on: bsockets/trunk/list.h
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/misc.c
===================================================================
--- bsockets/trunk/misc.c	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/misc.c	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,48 @@
+#include <windows.h>
+
+#include "misc.h"
+
+
+char _G_VERY_BAD_NEWS_[] = "\nSerious error, can't cope. Here is as much information as we can give...\n\tfile:%s line:%d errno:%d win32 err:%d\n";
+
+
+HANDLE make_thread (void *func, void *data) {
+
+	return CreateThread(NULL,0,func, data, 0, NULL);
+
+}
+
+/*"borrowed" from msdn*/
+int winsock_start() {
+	WORD wVersionRequested;
+	WSADATA wsaData;
+	int err;
+
+	wVersionRequested = MAKEWORD( 2, 2 );
+
+	err = WSAStartup( wVersionRequested, &wsaData );
+
+	if ( err != 0 ) {
+
+		return 1;
+	}
+
+	if ( LOBYTE( wsaData.wVersion ) != 2 ||
+			HIBYTE( wsaData.wVersion ) != 2 ) {
+
+		WSACleanup();
+		return 1;
+	}
+
+	return 0;
+}
+
+
+//im not going to think about overflow. if you want to wait more than 5
+//years or something like that, TS
+int timeval_to_millis(struct timeval *tv) {
+
+	return (tv->tv_sec)*1000;
+
+}
+


Property changes on: bsockets/trunk/misc.c
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/misc.h
===================================================================
--- bsockets/trunk/misc.h	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/misc.h	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,41 @@
+#ifndef _MISC_H_
+#define _MISC_H_
+
+#include <windows.h>
+#include <stdio.h>
+
+extern char _G_VERY_BAD_NEWS_[];
+
+#define VERY_BAD_NEWS fprintf(stderr,_G_VERY_BAD_NEWS_,__FILE__,__LINE__,errno,(int)GetLastError()); fflush(stderr);
+
+
+#define CHECK_(X) 		if (!(X)) {\
+							errno = unixify_wsaerr(GetLastError());\
+							out = -1;\
+							goto fail;\
+						}\
+
+#define CHECK(X,Y) 		if (!(X)) {\
+							if (Y != 0) {\
+								errno = Y;\
+							}\
+							out = -1;\
+							goto fail;\
+						}\
+
+#define ASSERT(X)		if (!(X)) {  fflush(stdout); VERY_BAD_NEWS ; _exit(1); }
+
+#define MUTEX_ACQUIRE(X)	ASSERT(WaitForSingleObject(X,INFINITE) == WAIT_OBJECT_0)
+#define MUTEX_RELEASE(X)	ASSERT(ReleaseMutex(X))
+
+
+
+int winsock_start();
+HANDLE make_thread(void*,void*);
+
+//todo -- if timeval not defined, give it something to work with
+
+int timeval_to_millis(struct timeval *);
+
+#endif
+


Property changes on: bsockets/trunk/misc.h
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/notes
===================================================================
--- bsockets/trunk/notes	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/notes	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,40 @@
+- make sure atomic/release doesnt cause other calls to block when socket is in blocking mode
+- make sure we set all events before closing them, otherwise blocking calls might block indefinitely, also find a way to show that the socket as closed
+- always release mutex before closing them
+- atomic/release should ensure the socket in question is open and not closing.
+- you cannot free a socket unless you own the list otherwise we cannot guarantee that mutexes inside atomic() still exist
+- make sure that socket error is reset after everytime it is looked at
+
+
+- check how many open reads we can have
+
+Ridiculous scenarios to think about
+
+1. User is select()ing while close() is called on one of the waiting descriptors elsewhere
+
+Why this is bad:
+The user could be waiting for a read operation, 
+
+Why it doesn't matter:
+This behavior is illegal (?)
+
+What will happen in bsock
+select will return -1, sockopt err will be set
+
+2. 
+
+
+
+sync issues
+-----------
+1. error array does not need syncronization because it may only be set by the 
+   event manager and read after event array indicates it is finished with socket
+
+2. we do not need to worry about closing a socket while there is a pending connect
+   which is queued. bconnect() will block until this is at least connected, so
+   any bclose() will block until the appropraite time.
+
+compatability issues
+--------------------
+1. openssl uses SO_*TIMEO flags. winsock has no support for this option, and implementing
+   it would force us to rip apart the existing code. is there any nice way to code.


Property changes on: bsockets/trunk/notes
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/select.c
===================================================================
--- bsockets/trunk/select.c	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/select.c	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,407 @@
+#include <windows.h>
+
+#include "bsocket.h"
+#include "event.h"
+#include "socket.h"
+#include "select.h"
+#include "misc.h"
+#include "io.h"
+
+void bsocket_raise(struct bsocket *b, int type, int active, struct socket_env *env ) {
+
+	struct _wait_list *wl;
+
+	wl = NULL;
+
+	switch (type) {
+
+		case IS_READABLE:
+			wl = b->read_wl;
+		break;
+
+		case IS_WRITABLE:
+			wl = b->write_wl;
+		break;
+
+		case IS_EXCEPTED:
+			wl = b->except_wl;
+		break;
+
+		default:
+
+			ASSERT(FALSE);
+		break;
+
+	}
+
+	ASSERT(wl != NULL);
+
+	if (active) {
+		wl_activate(wl,0);
+	} else {
+		wl_deactivate(wl);
+	}
+
+}
+
+void socket_raise(int fd, int type, int active, struct socket_env *env) {
+
+	struct bsocket **l;
+
+	l = bsocket_list_get(env);
+
+	if (l != NULL) {
+
+		if (l[fd] != NULL) {
+			bsocket_raise(l[fd],type,active,env);
+		}
+
+		bsocket_list_release(env);
+
+	}
+
+}
+
+void socket_set_err(int fd, int err, struct socket_env *env) {
+
+	struct bsocket **l;
+
+	l = bsocket_list_get(env);
+
+	if (l != NULL) {
+
+		if (l[fd] != NULL) {
+
+			l[fd]->err = err;
+
+		}
+
+		bsocket_list_release(env);
+
+	}
+
+}
+
+void socket_last_err(int fd, int *err, struct socket_env *env) {
+
+	struct bsocket **l;
+
+	l = bsocket_list_get(env);
+
+	if (l != NULL) {
+
+		if (l[fd] != NULL) {
+
+			*err = l[fd]->err;
+			l[fd]->err = 0;
+
+		}
+
+		bsocket_list_release(env);
+
+	}
+
+}
+
+//atomicity is provided by caller -- must own list
+void bsocket_exception(struct bsocket *b, int err, struct socket_env *env) {
+
+//	ASSERT(err != 0);
+
+	if (b->s != INVALID_SOCKET) {
+
+		b->err		= err;
+		b->closed	= TRUE;
+		b->eof		= TRUE;
+
+		bsocket_raise(b,IS_READABLE,TRUE,env);
+		bsocket_raise(b,IS_WRITABLE,TRUE,env);
+		bsocket_raise(b,IS_EXCEPTED,TRUE,env);
+
+		post_exception(b,err,env);
+
+		b->s = INVALID_SOCKET;
+
+	}
+
+
+}
+
+void socket_exception(int fd, int err, struct socket_env *env) {
+
+	struct bsocket **l;
+
+	l = bsocket_list_get(env);
+
+	if (l != NULL) {
+
+		if (l[fd] != NULL) {
+			bsocket_exception(l[fd],err,env);
+		}
+
+		bsocket_list_release(env);
+
+	}
+
+}
+
+int wait_until_group(struct bsocket **b, int *flag, int num, int timeout, struct socket_env *env) {
+
+	struct _wait_list **wl;
+
+	int i,j;
+	int out;
+	int r;
+
+	out = 0;
+
+	wl = (struct _wait_list **)  malloc(sizeof(struct _wait_list*)*(num+1));
+
+	CHECK(wl != NULL,ENOMEM);
+
+	for (i=0; i<num; i++) {
+
+		switch(flag[i]) {
+
+			case IS_READABLE:
+				wl[i] = b[i]->read_wl;
+			break;
+
+			case IS_WRITABLE:
+				wl[i] = b[i]->write_wl;
+			break;
+
+			case IS_EXCEPTED:
+				wl[i] = b[i]->except_wl;
+			break;
+
+			default:
+				ASSERT(FALSE);
+			break;
+
+		}
+
+		ASSERT(wl[i] != NULL);
+
+	}
+
+	for (i=0; i<num; i++) {
+
+		for (j=0; j<i; j++) {
+
+			ASSERT(wl[j] != wl[i]);
+		}
+
+	}
+
+	//todo == major sync issue here, fix next
+	r = wait_many_timeout(wl,num,timeout);
+
+	if (r == -1) {
+		ASSERT( (errno == ETIMEDOUT && timeout != -1)
+				||
+				(errno == ECLOSED)
+
+		);
+		out = -1;
+	} else {
+		out = r;
+	}
+
+
+	fail:
+
+	if (wl != NULL) {
+		free(wl);
+	}
+
+	return out;
+
+}
+
+int wait_until(struct bsocket *b, int flag, struct socket_env *env) {
+
+	struct bsocket *bl[1];
+	int fl[1];
+
+	bl[0] = b;
+	fl[0] = flag;
+
+	return wait_until_group(bl,fl,1,-1,env);
+
+}
+
+
+
+
+#define M_INSERT(X)		objects[count] = X;\
+						count++;\
+
+#define M_DELETE(X)		ASSERT(X < mutex_count);\
+						ASSERT(X >= 0);\
+						for (_i=X; _i<mutex_count-1; _i++) {\
+							desired_objects[_i] = desired_objects[_i+1];\
+						}\
+						mutex_count--;\
+
+
+int select_win32(int z, bfd_set *readfds, bfd_set *writefds, bfd_set *exceptfds,
+	struct timeval *timeout, struct socket_env *env) {
+
+	const int sets = 3;
+
+	bfd_set *fds[sets];
+	int		*t;
+
+
+	struct bsocket **b;
+	struct bsocket **l;
+	struct bsocket *c;
+
+	int count;
+	int i,j;
+	int fd_count;
+	int out;
+	int r;
+
+	int to;
+
+	int types[sets];
+
+	fds[0] = readfds;
+	fds[1] = writefds;
+	fds[2] = exceptfds;
+
+	types[0] = IS_READABLE;
+	types[1] = IS_WRITABLE;
+	types[2] = IS_EXCEPTED;
+
+
+	b = NULL;
+	fd_count = 0;
+	l = NULL;
+
+	//get this size
+	for (i=0; i<sets; i++) {
+		if (fds[i] != NULL) {
+			fd_count += fds[i]->count;
+		}
+	}
+
+	CHECK(fd_count > 0,EINVAL);
+
+
+	b = (struct bsocket**) malloc((sizeof(struct bsocket*))*(fd_count+1));
+
+	t = (int*) malloc((sizeof (int))*(fd_count+1));
+
+	CHECK(b != NULL,ENOMEM);
+	CHECK(t != NULL,ENOMEM);
+
+	l = bsocket_list_get(env);
+
+	CHECK(l != NULL,0);
+
+	count = 0;
+
+	for (i=0; i<sets; i++) {
+
+		if (fds[i] != NULL) {
+
+			for (j=0; j<fds[i]->count; j++) {
+
+				b[count] = l[fds[i]->list[j]];
+				t[count] = types[i];
+
+				count++;
+			}
+		}
+	}
+
+	bsocket_list_release(env);
+	l = NULL;
+
+	/*convert timeval to milliseconds*/
+	if (timeout != NULL) {
+		to = timeval_to_millis(timeout);
+	} else {
+		to = -1;
+	}
+
+	//todo -- major, what happens if someone closes socket right here
+	//(before we start waiting on it?)
+
+	r = wait_until_group(b,t,fd_count,to,env);
+
+	if (r != -1) {
+
+		//we need to know that socket is still valid
+		c = bsocket_get(b[r]->fd,AS_GLOBAL,env);
+		CHECK(c != NULL,0);
+
+		if (readfds != NULL) {
+			BFD_ZERO(readfds);
+		}
+
+		if (writefds != NULL) {
+			BFD_ZERO(writefds);
+		}
+
+		if (exceptfds != NULL) {
+			BFD_ZERO(exceptfds);
+		}
+
+		switch (t[r]) {
+
+			case IS_READABLE:
+				BFD_SET(c->fd,readfds);
+			break;
+
+			case IS_WRITABLE:
+				BFD_SET(c->fd,writefds);
+			break;
+
+			case IS_EXCEPTED:
+				BFD_SET(c->fd,exceptfds);
+			break;
+
+			default:
+				ASSERT(FALSE);
+			break;
+		}
+
+
+		bsocket_release(c->fd,AS_GLOBAL,env);
+
+		out = 1;
+
+	} else {
+
+		if (errno == ETIMEDOUT) {
+
+			out = 0;
+
+		} else {
+
+			out = -1;
+
+		}
+
+	}
+
+
+	fail:
+	if (b != NULL) {
+		free(b);
+	}
+
+	if (l != NULL) {
+		bsocket_list_release(env);
+	}
+
+	return out;
+
+}
+
+
+


Property changes on: bsockets/trunk/select.c
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/select.h
===================================================================
--- bsockets/trunk/select.h	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/select.h	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,29 @@
+#ifndef _SELECT_H_
+#define _SELECT_H_
+
+#include "socket.h"
+
+int  bsocket_last_err (struct bsocket *, int *, struct socket_env*);
+void bsocket_set_err (struct bsocket *, int, struct socket_env*);
+
+void socket_last_err(int, int*,struct socket_env*);
+void socket_set_err(int, int ,struct socket_env*);
+
+int wait_until_group(struct bsocket**, int*, int, int, struct socket_env*);
+int wait_until(struct bsocket *,int, struct socket_env*);
+
+
+void bsocket_raise(struct bsocket *, int, int, struct socket_env*);
+void bsocket_exception(struct bsocket *, int,  struct socket_env*);
+
+void socket_raise(int, int, int, struct socket_env*);
+void socket_exception(int, int, struct socket_env*);
+
+
+#define IS_READABLE		(1)
+#define IS_WRITABLE		(2)
+#define IS_EXCEPTED		(4)
+
+#endif
+
+


Property changes on: bsockets/trunk/select.h
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/socket.c
===================================================================
--- bsockets/trunk/socket.c	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/socket.c	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,510 @@
+#include <windows.h>
+#include <winbase.h>
+#include <stdio.h>
+#include <errno.h>
+
+#include "event.h"
+#include "bsocket.h"
+#include "errno.h"
+#include "socket.h"
+#include "unix.h"
+#include "list.h"
+#include "wait.h"
+#include "sync.h"
+#include "select.h"
+#include "misc.h"
+
+//support only turning off and on socket blocking, will add more features as needed
+int fcntl_win32(int fd, int cmd, long args, struct socket_env *env ) {
+
+	struct bsocket *b;
+	int out;
+
+	b = bsocket_get(fd,AS_GLOBAL,env);
+	CHECK(b != NULL,0);
+
+	out = 0;
+
+	switch (cmd) {
+
+		case F_SETFL:
+
+			switch(args) {
+
+				case O_NONBLOCK:
+					b->blocking = FALSE;
+				break;
+
+				case O_BLOCK:
+					b->blocking = TRUE;
+				break;
+
+				default:
+					CHECK(FALSE, ENOTSUP);
+				break;
+
+			}
+
+		break;
+
+		default:
+			CHECK(FALSE, ENOTSUP);
+		break;
+
+	}
+
+	fail:
+
+	if (b != NULL)
+		bsocket_release(fd,AS_GLOBAL,env);
+
+	return out;
+
+}
+
+//we will add more features as they become needed
+int getsockopt_win32(int fd, int level, int optname, void *optval, int *optlen, struct socket_env *env) {
+
+	struct bsocket **l;
+
+	int out;
+
+
+	l = NULL;
+	l = bsocket_list_get(env);
+
+	CHECK(l[fd] != NULL,EBADF);
+
+	out = 0;
+
+	switch (level) {
+
+		case SOL_SOCKET:
+
+			switch (optname) {
+
+				case SO_ERROR:
+
+
+					CHECK(*optlen >= sizeof(int),EINVAL);
+
+					*((int*) optval) = l[fd]->err;
+					*optlen = sizeof(int);
+
+
+				break;
+
+				default:
+
+				break;
+
+			}
+
+		break;
+
+		default:
+			CHECK(FALSE,ENOTSUP);
+		break;
+
+	}
+
+	fail:
+
+	bsocket_list_release(env);
+
+	return out;
+}
+
+
+int connect_win32(int fd, struct sockaddr *name, size_t len, struct socket_env *env) {
+
+	struct bsocket *b;
+
+	int err;
+	int out;
+
+	out = 0;
+
+	b = bsocket_get(fd,AS_RW,env);
+
+	CHECK(b != NULL,0);
+	CHECK(b->connected == FALSE,EISCONN);
+	CHECK(b->connecting == FALSE,EALREADY);
+
+	b->connecting = TRUE;
+
+	socket_raise(fd,IS_WRITABLE,FALSE,env);
+
+	out = post_connect(b,name, len, env);
+
+	if (b->blocking) {
+
+		wait_until(b,IS_WRITABLE,env);
+
+		socket_last_err(fd,&err,env);
+
+		if (err) {
+
+			errno 	=  err;
+			out 	= -1;
+
+		} else {
+
+			out = 0;
+
+		}
+
+	} else {
+
+		errno = EINPROGRESS;
+		out = -1;
+
+	}
+
+	fail:
+
+	if (b != NULL) {
+		bsocket_release(fd,AS_RW,env);
+	}
+
+
+	return out;
+
+}
+
+void bsocket_free(struct bsocket *b) {
+
+	if (b->read_wl != NULL) {
+		ASSERT(b->read_wl->waiting == NULL);
+		wl_free(b->read_wl);
+	}
+
+	if (b->write_wl != NULL) {
+		ASSERT(b->write_wl->waiting == NULL);
+		wl_free(b->write_wl);
+	}
+
+	if (b->except_wl != NULL) {
+		ASSERT(b->except_wl->waiting == NULL);
+		wl_free(b->except_wl);
+	}
+
+}
+
+
+//todo handle SO_LINGER
+int bsocket_close(struct bsocket *b, struct socket_env *env) {
+
+	SetEvent(b->close_e);
+
+	bsocket_exception(b,ECLOSED,env);
+
+	bsocket_list_release(env);
+
+	//manually aquire all socket mutexes (don't use atomic!)
+	MUTEX_ACQUIRE(b->read_m);
+	MUTEX_ACQUIRE(b->write_m);
+	MUTEX_ACQUIRE(env->list_m);
+
+	if (!b->closed) {
+		ASSERT(post_close(b,env) == 0);
+	}
+
+	//return fd
+	list_enqueue((void*) b->fd, env->free_q);
+
+	bsocket_free(b);
+
+	env->b[b->fd] = NULL;
+
+	//todo if there is an outstanding error, that becomes the output of close
+
+	return 0;
+
+}
+
+//todo -- fix so that close_win32 does not require any new memory
+int close_win32(int fd, struct socket_env *env) {
+
+	struct bsocket **l;
+
+	int out;
+
+	l = bsocket_list_get(env);
+	ASSERT(l != NULL);
+	CHECK(l[fd] != NULL,EBADF);
+
+	out = bsocket_close(l[fd],env);
+
+	bsocket_list_release(env);
+
+	fail:
+
+	return out;
+
+
+}
+
+struct bsocket *bsocket_new(SOCKET s) {
+
+	struct bsocket *b;
+
+	int out;
+
+	b = (struct bsocket *) malloc(sizeof(struct bsocket));
+
+	CHECK(b != NULL,ENOMEM);
+
+	b->s			= s;
+	b->blocking 	= TRUE;
+	b->connected 	= FALSE;
+	b->connecting	= FALSE;
+
+	b->read_m 		= NULL;
+	b->write_m 		= NULL;
+
+	b->close_e		= NULL;
+
+	b->read_wl		= NULL;
+	b->write_wl		= NULL;
+	b->except_wl	= NULL;
+
+	b->in_q			= NULL;
+	b->partner 		= NULL;
+
+	b->read_m = CreateMutex(NULL,FALSE,NULL);
+	CHECK_(b->read_m != NULL);
+
+	b->write_m = CreateMutex(NULL,FALSE,NULL);
+	CHECK_(b->write_m != NULL);
+
+	b->close_e = CreateEvent(NULL,TRUE,FALSE,NULL);
+	CHECK_(b->close_e != NULL);
+
+	b->err = 0;
+	b->connected = FALSE;
+	b->connecting = FALSE;
+	b->closed = FALSE;
+
+	b->read_wl	= wl_new();
+	CHECK(b->read_wl != NULL,0);
+	wl_activate(b->read_wl,0);
+
+	b->write_wl	= wl_new();
+	CHECK(b->write_wl != NULL,0);
+	wl_activate(b->write_wl,0);
+
+	b->except_wl = wl_new();
+	CHECK(b->except_wl != NULL, 0);
+	//do not open except list
+
+	b->write_buf = NULL;
+
+	b->in_q = list_new();
+	CHECK(b->in_q != NULL,0);
+
+	fail:
+
+	if (out == -1) {
+
+		bsocket_free(b);
+		return NULL;
+
+	} else {
+		return b;
+	}
+
+	ASSERT(FALSE);
+}
+
+
+int socket_win32(int af, int type, int protocol, struct socket_env *env) {
+
+	SOCKET s;
+
+	struct bsocket *b;
+	int nonblocking;
+	int fd;
+	int out;
+
+	nonblocking=1;
+	out = 0;
+
+	s = WSASocket(af,type,protocol,NULL,0,WSA_FLAG_OVERLAPPED);
+
+	CHECK_(s != INVALID_SOCKET);
+
+	CHECK_( ioctlsocket(s,FIONBIO,(u_long*) &nonblocking) == 0);
+
+	b = bsocket_new(s);
+
+	CHECK(b != NULL,0);
+
+	CHECK( (fd = claim_free_fd(b,env)) >= 0,0);
+
+	b->fd = fd;
+
+	fail:
+
+	if (out == -1) {
+
+		if (b != NULL) {
+			bsocket_free(b);
+		}
+
+		if (s != INVALID_SOCKET) {
+			ASSERT(closesocket(s) == 0);
+		}
+
+	} else {
+		out = fd;
+
+	}
+
+	return out;
+
+}
+
+int socket_init_win32(struct socket_env *env) {
+
+	HANDLE h;
+
+	int i;
+	int out;
+
+	out = 0;
+	h = NULL;
+
+	ASSERT( sizeof(int) == sizeof(DWORD));
+
+
+	if (env == NULL) {
+
+		env = (struct socket_env *)
+				malloc(sizeof(struct socket_env));
+		__GLOBAL_BSOCKET_ENV_ = env;
+	}
+
+	CHECK(env != NULL,ENOMEM);
+
+	env->free_q = list_new();
+	CHECK(env->free_q != NULL,0);
+
+	env->event_q = list_new();
+	CHECK(env->event_q != NULL,0);
+
+	for (i=0; i<MAX_BSOCKETS; i++) {
+
+		env->b[i] = NULL;
+
+		CHECK(list_enqueue((void*) i,env->free_q) != NULL,0);
+
+	}
+
+	env->post_e = WSACreateEvent();
+	CHECK_(env->post_e != WSA_INVALID_EVENT);
+
+	env->shutdown_e = CreateEvent(NULL,TRUE,FALSE,NULL);
+	CHECK_(env->shutdown_e != NULL);
+
+	env->post_m = CreateMutex(NULL,FALSE,NULL);
+	CHECK_(env->post_m != NULL);
+
+	env->list_m = CreateMutex(NULL,FALSE,NULL);
+	CHECK_(env->list_m != NULL);
+
+
+	env->wait_m = CreateMutex(NULL,FALSE,NULL);
+	CHECK_(env->wait_m != NULL);
+
+	CHECK_((h = make_thread(event_manager,env)) != NULL);
+
+	env->event_t = h;
+
+	event_ping(env);
+
+	fail:
+	//todo fix memory leak on failure
+	if (h != NULL && out == -1) {
+		TerminateThread(h,1);
+		CloseHandle(h);
+	}
+
+	return out;
+
+}
+
+int socket_cleanup_win32(struct socket_env *env) {
+
+	struct bsocket **l;
+
+	int i;
+	int out;
+	int r;
+	int was_null;
+
+	out = 0;
+	was_null = FALSE;
+
+	//todo -- i can smell a sync issue here
+
+	if (env == NULL) {
+		env = __GLOBAL_BSOCKET_ENV_;
+		was_null = TRUE;
+
+	}
+
+	ASSERT(SetEvent(env->shutdown_e));
+
+	MUTEX_ACQUIRE(env->list_m);
+
+	l = env->b;
+
+	for (i=0; i<MAX_BSOCKETS; i++) {
+
+		if (l[i] != NULL) {
+
+			r = bsocket_close(l[i],env);
+
+		}
+
+	}
+
+	//todo replace with post_shutdown
+	ASSERT(event_post(NULL,EV_SHUTDOWN,NULL,NULL,NULL,FALSE,env) == 0);
+
+
+	r = WaitForSingleObject (env->event_t,1000);
+
+	ASSERT(r == WAIT_OBJECT_0);
+
+
+	//make sure we have cleaned up properly
+	//todo - get rid of this when we are confident it isn't a problem
+	for (i=0; i<MAX_BSOCKETS; i++) {
+
+		ASSERT(env->b[i] == NULL);
+	}
+
+
+	ASSERT(CloseHandle(env->shutdown_e));
+	//ASSERT(CloseHandle(env->free_m));
+	ASSERT(CloseHandle(env->list_m));
+	ASSERT(CloseHandle(env->post_m));
+	ASSERT(CloseHandle(env->wait_m));
+
+	list_free(env->free_q);
+
+	free(env);
+
+	if (was_null) {
+		__GLOBAL_BSOCKET_ENV_ = NULL;
+	}
+
+	goto fail;
+	fail:
+
+	return out;
+
+}
+


Property changes on: bsockets/trunk/socket.c
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/socket.h
===================================================================
--- bsockets/trunk/socket.h	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/socket.h	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,92 @@
+#ifndef _SOCKET_H_
+#define _SOCKET_H_
+
+#include <windows.h>
+#include <errno.h>
+#include <stdio.h>
+
+#include "unix.h"
+#include "bsocket.h"
+#include "sync.h"
+#include "wait.h"
+
+//todo move this checking stuff into another header
+
+#define AS_READ  		(1)
+#define AS_WRITE 		(2)
+#define AS_ERR			(4)
+
+#define AS_RW			(AS_READ|AS_WRITE)
+#define AS_GLOBAL		(AS_READ|AS_WRITE|AS_ERR)
+
+#define AE_POST 		(1)
+#define AE_LIST			(2)
+#define AE_FREE			(4)
+#define AE_WAIT			(8)
+
+#define AE_GLOBAL		(AE_POST|AE_LIST|AE_FREE|AE_WAIT)
+
+
+struct bsocket {
+
+	SOCKET s;
+
+	HANDLE read_m;
+	HANDLE write_m;
+
+	HANDLE close_e;
+
+	struct _wait_list	*read_wl;
+	struct _wait_list	*write_wl;
+	struct _wait_list	*except_wl;
+
+	int blocking;
+
+	int connected;
+	int connecting;
+	int closed;
+
+	int err;
+	int eof;
+
+	int fd;
+
+	/*writing business*/
+
+	WSABUF	write_wb;
+
+	void *write_buf;
+	void *write_pointer;
+	int write_len;
+
+	/*reading business*/
+	struct _list *in_q;
+
+	/*this is non-null if the socket is part of a socketpair*/
+	struct bsocket *partner;
+
+
+};
+
+struct socket_env {
+
+	HANDLE		shutdown_e;
+
+	HANDLE		list_m;
+	HANDLE		post_m;
+	//HANDLE		free_m;
+	HANDLE		wait_m;
+
+	HANDLE		event_t;
+
+	WSAEVENT	post_e;
+
+	struct bsocket *b[MAX_BSOCKETS];
+
+	struct _list *free_q;
+	struct _list *event_q;
+
+};
+
+
+#endif


Property changes on: bsockets/trunk/socket.h
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/sync.c
===================================================================
--- bsockets/trunk/sync.c	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/sync.c	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,296 @@
+#include <windows.h>
+#include <errno.h>
+
+#include "unix.h"
+#include "socket.h"
+#include "sync.h"
+#include "list.h"
+#include "misc.h"
+
+int claim_free_fd(struct bsocket *b, struct socket_env *env) {
+
+	struct bsocket **l;
+
+	int out;
+
+	//we don't care about the output, this just gives us list
+	//ownership
+	l = bsocket_list_get(env);
+
+	CHECK(l != NULL, 0);
+
+	if (list_dequeue(&out,env->free_q) == 0) {
+
+		l[out] = b;
+
+	} else {
+		errno = EMFILE;
+		out = -1;
+	}
+
+	bsocket_list_release(env);
+
+	fail:
+	return out;
+}
+
+
+struct bsocket **bsocket_list_get(struct socket_env *env) {
+
+	if (atomic(-1,AE_LIST)) {
+		return NULL;
+	}
+
+
+	return env->b;
+
+}
+
+void bsocket_list_release(struct socket_env *env) {
+	release(-1,AE_LIST);
+}
+
+int wait_lock(struct socket_env *env) {
+
+	//todo is there any acceptable reason for this to fail?
+	ASSERT(atomic(-1,AE_WAIT) == 0);
+
+	return 0;
+
+}
+
+void wait_lock_release(struct socket_env *env) {
+
+	release(-1,AE_WAIT);
+
+}
+
+struct bsocket *bsocket_get(int fd, int access, struct socket_env *env) {
+
+	struct bsocket **l;
+	struct bsocket *out;
+
+	if (fd < 0) {
+		errno = EINVAL;
+		return NULL;
+	}
+
+	if (fd >= MAX_BSOCKETS) {
+		errno = EINVAL;
+		return NULL;
+	}
+
+	out = NULL;
+	l = bsocket_list_get(env);
+
+	if (l != NULL) {
+
+		if (!atomic(fd,access)) {
+			out = l[fd];
+		}
+
+		bsocket_list_release(env);
+
+	}
+
+//	printf("bsocket_get out\n");
+	//fflush(stdout);
+
+	return out;
+
+
+}
+
+void bsocket_release(int fd, int access, struct socket_env *env) {
+
+	release(fd,access);
+
+}
+
+#define M_INSERT(X)		objects[count] = X;\
+						count++;\
+
+#define M_DELETE(X)		ASSERT(X < mutex_count);\
+						ASSERT(X >= 0);\
+						for (_i=X; _i<mutex_count-1; _i++) {\
+							desired_objects[_i] = desired_objects[_i+1];\
+						}\
+						mutex_count--;\
+
+//in this version, we leave it up to the calling thread to have appropriate lock ownership
+//undefined behavior occurs if this function is called after an environment shutdown
+int lookup_mutexes(int fd, int type, struct socket_env *env, HANDLE *objects) {
+
+	struct bsocket *b;
+	int count;
+
+	count = 0;
+
+	if (fd == -1) {
+
+		M_INSERT(env->shutdown_e)
+
+		if (type & AE_POST) {
+			M_INSERT(env->post_m);
+
+		}
+
+		if (type & AE_LIST) {
+			M_INSERT(env->list_m);
+		}
+
+	//	if (type & AE_FREE) {
+	//		M_INSERT(env->free_m);
+	//	}
+
+		if (type & AE_WAIT) {
+			M_INSERT(env->wait_m);
+		}
+
+
+	} else {
+
+		//to look up these mutexes, and guarantee they will not be closed during
+		//lookup the calling thread must own one mutex
+
+		b = env->b[fd];
+
+		if (b == NULL) {
+			errno = EBADF;
+			return -1;
+		}
+
+		M_INSERT(b->close_e)
+
+		if (type & AS_READ) {
+
+			M_INSERT(b->read_m);
+		}
+
+		if (type & AS_WRITE) {
+
+			M_INSERT(b->write_m);
+		}
+
+		//if (type & AS_ERR) {
+		//	M_INSERT(b->err_m);
+		//}
+
+
+	}
+
+
+	return count;
+}
+
+//claim ownership of the mutexes listed in type belonging to fd, or the global
+//environment when fd = -1
+int atomic_win32(int fd, int type, struct socket_env *env) {
+
+	int _i;
+
+	HANDLE close_event;
+	HANDLE desired_objects[8];
+
+	int mutex_count,mutex_count_2 = 0;
+	int out;
+
+	int r;
+
+	out = 0;
+
+	r =  lookup_mutexes(fd,type,env,desired_objects);
+
+	if (r == -1) {
+
+		return -1;
+
+	} else {
+		mutex_count = r;
+	}
+
+
+	mutex_count_2 = mutex_count;
+
+	close_event = desired_objects[0];
+
+	ASSERT(mutex_count > 1);
+
+	while (mutex_count > 1) {
+
+	SetLastError(0);
+
+	//todo -- get rid of this
+	r = WaitForMultipleObjects(mutex_count,desired_objects,FALSE,INFINITE);
+
+		switch(r) {
+
+			case WAIT_TIMEOUT:
+			//this might not seem like a critical failure, however i want to know
+			//if we are not releasing mutexes
+			case WAIT_ABANDONED:
+			//there is no acceptable reason for wfmo() to fail
+			case WAIT_FAILED:
+				ASSERT(FALSE);
+			break;
+
+			//close triggered, we should leave immediately
+			case WAIT_OBJECT_0:
+				//not a critical issue but we want to know if we claimed any mutexes
+				//(we shouldn't have)
+				ASSERT(mutex_count_2 == mutex_count);
+				errno = ECLOSED;
+				mutex_count = 0;
+				out = -1;
+			break;
+
+			default:
+				M_DELETE(r - WAIT_OBJECT_0);
+
+			break;
+
+		}
+	}
+
+
+	if (out != -1) {
+
+		ASSERT(mutex_count == 1);
+		ASSERT(desired_objects[0] == close_event);
+	}
+
+	return out;
+
+}
+
+void release_win32(int fd, int type, struct socket_env *env) {
+
+	HANDLE desired_events[8];
+
+	int r;
+
+	//on the first iteration of shutdown, it is closing the socket and failing
+	//because it thinks it doesnt exist because it isn't in the list
+
+	r = lookup_mutexes(fd,type,env,desired_events);
+
+	//dont use this with type as zero
+	ASSERT( r > 1);
+	//printf("start ",r);
+
+	//rember events[0] is close event, we dont want to touch that yet
+	while (r > 1) {
+		/*todo assert around this*/
+	//	printf("%d ",desired_events[r-1]);
+	//	fflush(stdout);
+		ASSERT(ReleaseMutex(desired_events[r-1]));
+		r--;
+	}
+
+	//printf("\n");
+	//fflush(stdout);
+
+
+}
+
+


Property changes on: bsockets/trunk/sync.c
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/sync.h
===================================================================
--- bsockets/trunk/sync.h	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/sync.h	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,24 @@
+#ifndef _SYNC_H_
+#define _SYNC_H_
+
+#define	SYNC_PRIVATE_ERRNO 120000
+
+
+#define atomic(X,Y)		atomic_win32(X,Y,env)
+#define release(X,Y)	release_win32(X,Y,env)
+
+struct bsocket **bsocket_list_get(struct socket_env*);
+void bsocket_list_release(struct socket_env*);
+
+struct bsocket *bsocket_get(int, int, struct socket_env *);
+void bsocket_release(int, int, struct socket_env *);
+
+int atomic_win32(int, int, struct socket_env *);
+void release_win32(int, int, struct socket_env *);
+
+int claim_free_fd(struct bsocket *, struct socket_env *);
+void release_fd(int, struct socket_env *);
+
+
+#endif
+


Property changes on: bsockets/trunk/sync.h
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/test.c
===================================================================
--- bsockets/trunk/test.c	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/test.c	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,1769 @@
+#include <windows.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <time.h>
+#include <errno.h>
+
+#include "bsocket.h"
+#include "socket.h"
+#include "select.h"
+
+#include "list.h"
+#include "test.h"
+#include "unix.h"
+#include "sync.h"
+#include "wait.h"
+#include "misc.h"
+
+#define TEST_MSG "hello!"
+
+int res;
+
+//test ideas
+/*
+1. multiple non-blocking connect()s on same socket (should fail with EAGAIN?)
+2. if socket is connected, try connecting (should fail with ECONN)
+3. is there some case where we can add a connect event and the event manager wont think we've
+   added it?
+4. assert event_q is empty when exiting event manager
+5. test select timeout
+*/
+
+
+//todo is there something that is guarnateed to be unreachable?
+#define BAD_ADDR "1.2.3.4"
+
+struct sockaddr_in localhost;
+struct sockaddr_in google;
+
+struct test_case {
+	int (*fun)();
+	void *data;
+	char *desc;
+
+};
+
+HANDLE new_thread(void *fun, void *data) {
+	return CreateThread(NULL,0,fun,data,0,NULL);
+}
+
+
+static int ran(int max) {
+
+	int out;
+
+	//todo might cause overflow if RAND_MAX is too big on your system
+	out = ((rand()*max)-1)/(RAND_MAX);
+
+	ASSERT(out < max);
+	ASSERT(out >= 0);
+
+	return out;
+
+}
+
+
+void loopback_server_helper(SOCKET s) {
+
+	int c;
+
+
+	while ( (c = accept(s,NULL,0)) != INVALID_SOCKET) {
+		//
+	}
+
+	closesocket(s);
+
+}
+
+int loopback_server(int *port) {
+
+	SOCKET s;
+
+	struct sockaddr_in in;
+	struct sockaddr_in name;
+
+	int out;
+	int r;
+	int len;
+	out = 0;
+
+	s = socket(AF_INET,SOCK_STREAM,0);
+
+	CHECK_(s != INVALID_SOCKET);
+
+	in.sin_family		 	= AF_INET;
+	in.sin_addr.s_addr	  	= htonl(INADDR_LOOPBACK);
+	in.sin_port				= htons(0);
+
+	r = bind(s,(struct sockaddr*) &in,sizeof(in));
+
+	CHECK_(r != SOCKET_ERROR);
+
+	r = listen(s,10);
+
+	CHECK_(r != SOCKET_ERROR);
+
+	len = sizeof(struct sockaddr_in);
+	r = getsockname(s, (struct sockaddr*) &name, &len);
+
+	CHECK_(r != SOCKET_ERROR);
+
+	CHECK_(new_thread(loopback_server_helper,(void*) s) != NULL);
+
+	*port = name.sin_port;
+	out = (int) s;
+
+	fail:
+
+	if (out == -1 && s != INVALID_SOCKET ) {
+
+	}
+
+	return out;
+}
+
+
+int test_active_refusal_nonblocking(){
+
+	bfd_set fds;
+
+	int fd;
+	int r;
+
+	int z;
+	int len;
+
+	TEST(bsocket_init(NULL) == 0);
+
+	fd = bsocket(AF_INET,SOCK_STREAM,0);
+	TEST(fd == 0);
+
+	//you better not be running a local service on port 81
+	bfcntl(fd,F_SETFL,O_NONBLOCK);
+
+	r = bconnect(fd,(struct sockaddr*)&localhost,sizeof(localhost));
+
+	TEST(r == -1);
+
+	TEST(errno == EINPROGRESS);
+
+	BFD_ZERO(&fds);
+	TEST(fds.count ==  0);
+
+	BFD_SET(fd,&fds);
+
+	TEST(fds.count == 1);
+
+	r = bselect(0,NULL,&fds,NULL,NULL);
+
+	TEST(r == 1);
+
+	TEST(BFD_ISSET(fd,&fds));
+
+	len = sizeof(z);
+	//printf("%d\n",1);
+	//fflush(stdout);
+
+	TEST(bgetsockopt(fd,SOL_SOCKET,SO_ERROR,&z,&len) == 0);
+
+	TEST(z == ECONNREFUSED);
+
+
+
+	TEST(bsocket_shutdown(NULL) == 0);
+
+	return 0;
+}
+
+
+int test_active_refusal_blocking() {
+
+	int fd;
+	int r;
+
+	TEST(bsocket_init(NULL) == 0);
+
+	fd = bsocket(AF_INET,SOCK_STREAM,0);
+
+	TEST(fd == 0);
+
+	errno = 0;
+	r = bconnect(fd, (struct sockaddr*) &localhost,sizeof(localhost));
+
+	TEST(r == -1);
+
+	TEST(errno == ECONNREFUSED);
+
+	TEST(bsocket_shutdown(NULL) == 0);
+
+	return  0;
+}
+
+int _g_fd_;
+
+int test_connect_close_helper(struct sockaddr_in *name) {
+
+	int r;
+
+	r = bconnect(_g_fd_,(struct sockaddr*)name,sizeof(struct sockaddr_in));
+
+	SILENT_TEST(r == -1);
+	SILENT_TEST(errno == ECLOSED);
+
+	return 0;
+
+}
+
+int test_connect_close() {
+
+	HANDLE h;
+
+	struct sockaddr_in in;
+	int fd;
+	int r;
+
+	TEST(bsocket_init(NULL) == 0);
+
+	fd = bsocket(AF_INET,SOCK_STREAM,0);
+
+	TEST(fd == 0);
+
+	in.sin_family = AF_INET;
+	in.sin_addr.s_addr = inet_addr(BAD_ADDR);
+	in.sin_port = htons(10001);
+
+	_g_fd_ = fd;
+
+	h = new_thread(test_connect_close_helper,&in);
+
+	Sleep(5);
+
+	TEST(bclose(fd) == 0);
+
+	TEST(WaitForSingleObject(h,100) == 0);
+	TEST(GetExitCodeThread(h,(DWORD*) &r));
+
+	TEST(r == 0);
+
+	TEST(bsocket_shutdown(NULL) == 0);
+
+	return 0;
+
+}
+
+
+int test_parallel_connect() {
+
+	struct sockaddr_in in;
+
+	int fd;
+	int r;
+
+	TEST(bsocket_init(NULL) == 0);
+
+	fd = bsocket(AF_INET,SOCK_STREAM,0);
+
+	TEST(fd == 0);
+
+	in.sin_family = AF_INET;
+	in.sin_addr.s_addr = inet_addr(BAD_ADDR);
+	in.sin_port = htons(10001);
+
+	bfcntl(fd,F_SETFL,O_NONBLOCK);
+
+	errno = 0;
+
+	r = bconnect(fd,(struct sockaddr*) &in, sizeof(in));
+
+	TEST(r == -1);
+	TEST(errno == EINPROGRESS);
+
+	r = bconnect(fd,(struct sockaddr*) &in, sizeof(in));
+
+
+	TEST(r == -1);
+	TEST(errno == EALREADY);
+
+	bfcntl(fd,F_SETFL,O_BLOCK);
+
+	r = bconnect(fd,(struct sockaddr*) &in, sizeof(in));
+
+	TEST(r == -1);
+	TEST(errno == EALREADY);
+
+	fd = bsocket(AF_INET,SOCK_STREAM,0);
+
+	TEST(fd >= 0);
+
+	google.sin_port = htons(80);
+
+	r = bconnect(fd,(struct sockaddr*) &google, sizeof(google));
+
+	TEST(r == 0);
+
+	r = bconnect(fd,(struct sockaddr*)  &google, sizeof(google));
+
+	TEST(r == -1);
+	TEST(errno == EISCONN);
+
+
+	TEST(bsocket_shutdown(NULL) == 0);
+
+	return 0;
+}
+
+int test_badconnection_blocking() {
+
+	int fd;
+	int r;
+
+	struct sockaddr_in in;
+
+	TEST(bsocket_init(NULL) == 0);
+
+	fd = bsocket(AF_INET,SOCK_STREAM,0);
+
+	TEST(fd >= 0);
+	TEST(fd < MAX_BSOCKETS);
+
+	in.sin_family = AF_INET;
+	in.sin_addr.s_addr = inet_addr(BAD_ADDR);
+	in.sin_port = htons(80);
+
+	errno = 0;
+	r = bconnect(fd,(struct sockaddr*) &in,sizeof(in));
+
+	TEST( r == -1);
+	TEST( errno == ETIMEDOUT);
+
+
+	TEST(bsocket_shutdown(NULL) == 0);
+
+	return 0;
+
+}
+
+int test_overflowsocket() {
+
+	int fd[MAX_BSOCKETS];
+
+	int f;
+	int i;
+
+	TEST(bsocket_init(NULL) == 0);
+
+	for (i=0; i<MAX_BSOCKETS; i++) {
+
+		fd[i] = bsocket(AF_INET,SOCK_STREAM,0);
+
+		TEST(fd[i] >= 0);
+		TEST(fd[i] < MAX_BSOCKETS);
+
+	}
+
+	errno = 0;
+
+	f = bsocket(AF_INET,SOCK_STREAM,0);
+
+	TEST(f == -1);
+	TEST(errno == EMFILE);
+
+	errno = 0;
+
+
+
+
+	TEST(bclose(fd[0]) == 0);
+	TEST(errno == 0);
+
+
+
+	f = bsocket(AF_INET,SOCK_STREAM,0);
+
+
+	TEST(f != -1);
+	TEST(errno == 0);
+
+	for (i=1; i<MAX_BSOCKETS; i++)	{
+
+		TEST(bclose(fd[i]) == 0);
+		TEST(errno == 0);
+
+	}
+
+
+
+
+	f = bsocket(AF_INET,SOCK_STREAM,0);
+
+	TEST(f != -1);
+
+	TEST(bsocket_shutdown(NULL) == 0);
+
+	return 0;
+
+}
+
+//todo -- think of a way to check if there is a memory leak
+int test_socketinit() {
+
+	int fd;
+	int tests = 1;
+
+	while (tests--) {
+
+		TEST(bsocket_init(NULL) == 0 );
+
+		fd = bsocket(AF_INET,SOCK_STREAM,0);
+
+		TEST(fd >= 0);
+		TEST(fd < MAX_BSOCKETS);
+
+		TEST(bclose(fd) == 0);
+
+		TEST(bsocket_shutdown(NULL) == 0 );
+
+	}
+
+	return 0;
+}
+
+int test_createlist() {
+
+	struct _list *list;
+
+	list = list_new();
+
+	TEST(list != NULL);
+	TEST(list->head == NULL);
+	TEST(list->tail == NULL);
+
+	list_free(list);
+
+	return 0;
+
+}
+
+int test_basicqueue() {
+
+	struct _list *list;
+
+	int d = 666;
+	int r, res2;
+
+	list = list_new();
+
+	TEST(list != NULL);
+
+	TEST(list_enqueue((void*)d,list) != NULL);
+
+	TEST(list->head != NULL);
+	TEST(list->tail != NULL);
+	TEST(list->head == list->tail);
+
+	r = list_dequeue(&res2,list);
+
+	TEST(r == 0);
+	TEST(res2 == d);
+
+	TEST(list->head == NULL);
+	TEST(list->tail == NULL);
+
+	list_free(list);
+
+	return 0;
+
+}
+
+
+/*add random values to a queue, and them pull them out*/
+int test_bigqueue() {
+
+	const int num = 10001;
+
+	struct _list *list;
+
+	int *v = (int*) malloc((sizeof(int))*num);
+
+	int i;
+	int r;
+	int n;
+
+	list = list_new();
+
+	for (i=0; i<num; i++) {
+
+		r = rand();
+		v[i] = r;
+		TEST(list_enqueue((void*) r,list) != NULL);
+
+	}
+
+	for (i=0; i<num; i++) {
+		n = list_dequeue(&r,list);
+		TEST(n == 0);
+		TEST(r == v[i]);
+	}
+
+	n = list_dequeue(&r,list);
+	TEST(n != 0);
+
+	TEST(list->head == NULL);
+	TEST(list->tail == NULL);
+
+	free(v);
+	list_free(list);
+
+	return 0;
+
+}
+
+int test_envmutex_helper(struct socket_env *env) {
+
+	int claimed;
+
+	claimed = 0;
+
+	if (WaitForSingleObject(env->list_m,0) == WAIT_OBJECT_0) {
+		claimed++;
+	}
+
+
+//	if (WaitForSingleObject(env->free_m,0) == WAIT_OBJECT_0) {
+//		claimed++;
+//	}
+
+	if (WaitForSingleObject(env->post_m,0) == WAIT_OBJECT_0) {
+		claimed++;
+	}
+
+	if (WaitForSingleObject(env->wait_m,0) == WAIT_OBJECT_0) {
+		claimed++;
+	}
+
+	if (claimed) {
+		release(-1,AE_GLOBAL);
+	}
+
+	return claimed;
+
+}
+
+//todo - should do test for each mutex, but this is good enough for now
+int test_envmutex() {
+
+	HANDLE h;
+
+	struct socket_env *env;
+
+	int r;
+	int tests = 1000;
+
+	TEST((bsocket_init(NULL) == 0));
+
+	env = __GLOBAL_BSOCKET_ENV_;
+
+	while(tests--) {
+
+		atomic(-1,AE_GLOBAL);
+
+		/*check if any of the mutexes can be claimed (very bad if)*/
+		h = new_thread(test_envmutex_helper,env);
+
+		TEST(h != NULL);
+
+		r = WaitForSingleObject(h,1000);
+
+		TEST(r == WAIT_OBJECT_0);
+		TEST(GetExitCodeThread(h,(DWORD*)&r));
+
+		TEST(r == 0);
+		CloseHandle(h);
+
+		release(-1,AE_GLOBAL);
+
+		/*now make sure they were released properly*/
+		h = new_thread(test_envmutex_helper,env);
+
+		TEST(h != NULL);
+
+		r = WaitForSingleObject(h,1000);
+
+		TEST(r == WAIT_OBJECT_0);
+
+		TEST(GetExitCodeThread(h,(DWORD*)&r));
+
+		TEST(r == 3);
+
+		CloseHandle(h);
+
+	}
+
+	TEST(bsocket_shutdown(NULL) == 0);
+
+	return 0;
+
+}
+
+int test_envmutexclose_helper(struct socket_env *env) {
+
+	struct bsocket **l;
+
+
+	l = bsocket_list_get(env);
+
+	SILENT_TEST(l == NULL);
+	SILENT_TEST(errno == ECLOSED);
+
+	return 0;
+}
+
+int test_envmutexclose() {
+
+	HANDLE h;
+
+	struct bsocket **l;
+
+	struct socket_env *env;
+
+	int r;
+
+	TEST(bsocket_init(NULL) == 0);
+	env = __GLOBAL_BSOCKET_ENV_;
+
+	l = bsocket_list_get(env);
+	TEST(l != NULL);
+
+	h = new_thread(test_envmutexclose_helper,env);
+	Sleep(10);
+
+	TEST(bsocket_shutdown(NULL) == 0);
+
+	TEST(WaitForSingleObject(h,100) == WAIT_OBJECT_0);
+	TEST(GetExitCodeThread(h,(DWORD*) &r));
+
+	TEST(r == 0);
+
+	CloseHandle(h);
+
+	return 0;
+
+}
+
+int test_socketmutex_helper(struct bsocket *b) {
+
+	struct socket_env *env;
+	int count;
+
+	count = 0;
+
+	if (WaitForSingleObject(b->read_m,0) == WAIT_OBJECT_0)  {
+		count++;
+	}
+
+	if (WaitForSingleObject(b->write_m,0) == WAIT_OBJECT_0) {
+		count++;
+	}
+
+	if (count) {
+		env = __GLOBAL_BSOCKET_ENV_;
+		release(b->fd,AS_GLOBAL);
+	}
+
+	return count;
+}
+
+//todo - a test for each mutex individually
+int test_socketmutex() {
+
+	HANDLE h;
+
+	struct socket_env *env;
+
+	int r;
+	int fd;
+
+	TEST(bsocket_init(NULL) == 0);
+	env = __GLOBAL_BSOCKET_ENV_;
+
+	fd = bsocket(AF_INET,SOCK_STREAM,0);
+	TEST(fd == 0);
+
+	errno = 0;
+
+	r = atomic(fd,AS_GLOBAL);
+
+	TEST(r == 0);
+
+	h = new_thread(test_socketmutex_helper,env->b[fd]);
+	TEST(h != NULL);
+
+	r = WaitForSingleObject(h,2000);
+	TEST(r == WAIT_OBJECT_0);
+
+	GetExitCodeThread(h, (DWORD*) &r);
+	TEST(r == 0);
+
+	CloseHandle(h);
+
+	release(fd,AS_GLOBAL);
+
+	h = new_thread(test_socketmutex_helper,env->b[fd]);
+	TEST(h != NULL);
+
+	r = WaitForSingleObject(h,2000);
+	TEST(r == WAIT_OBJECT_0);
+
+	GetExitCodeThread(h, (DWORD*) &r);
+
+	TEST(atomic(fd,AS_GLOBAL) == 0);
+	release(fd,AS_GLOBAL);
+
+	TEST(r == 2);
+
+	CloseHandle(h);
+
+	TEST(bclose(fd) == 0);
+
+
+	TEST(bsocket_shutdown(NULL) == 0);
+
+	return 0;
+}
+
+
+//attempt to grab a mutex for a socket that does not exist
+int test_badsocket() {
+
+	struct socket_env *env;
+
+	int r;
+
+	TEST(bsocket_init(NULL) == 0);
+	env = __GLOBAL_BSOCKET_ENV_;
+
+	errno = 0;
+
+	r = atomic(0,AS_READ);
+
+	TEST(r == -1);
+	TEST(errno == EBADF);
+
+	TEST(bsocket_shutdown(NULL) == 0);
+
+	return 0;
+
+}
+
+int test_socketmutexclose_helper(int fd) {
+
+	struct socket_env *env;
+
+	int r;
+
+	env = __GLOBAL_BSOCKET_ENV_;
+
+	errno = 0;
+	r = atomic(fd,AS_GLOBAL);
+
+	SILENT_TEST(r == -1);
+	SILENT_TEST(errno == ECLOSED);
+
+
+	return 0;
+
+
+}
+
+int test_socketmutexclose() {
+
+	HANDLE h;
+	struct socket_env *env;
+
+	int fd;
+	int r;
+
+	TEST(bsocket_init(NULL) == 0);
+
+	fd = bsocket(AF_INET,SOCK_STREAM,0);
+
+	TEST(fd == 0);
+
+	env = __GLOBAL_BSOCKET_ENV_;
+
+	atomic(fd,AS_GLOBAL);
+
+	h = new_thread(test_socketmutexclose_helper,(void*)fd);
+
+	Sleep(30);
+
+	TEST(bclose(fd) == 0);
+
+	r = WaitForSingleObject(h,10000);
+
+	TEST(r == WAIT_OBJECT_0);
+
+	GetExitCodeThread(h,(DWORD*)&r);
+
+	TEST(r == 0);
+
+	CloseHandle(h);
+
+	TEST(bsocket_shutdown(NULL) == 0);
+
+	return 0;
+
+}
+
+int test_syncstruct() {
+
+	struct bsocket *b;
+	//struct bsocket **l;
+	struct socket_env *env;
+
+	int fd;
+	int rounds;
+
+	rounds = 1000;
+
+	TEST(bsocket_init(NULL) == 0);
+
+	fd = bsocket(AF_INET,SOCK_STREAM,0);
+	env = __GLOBAL_BSOCKET_ENV_;
+
+	TEST(fd == 0);
+
+	while(rounds--) {
+
+		b = bsocket_get(fd,AS_GLOBAL,env);
+
+		bsocket_release(fd,AS_GLOBAL,env);
+
+		TEST(bsocket_list_get(env) != NULL);
+
+		bsocket_list_release(env);
+
+	}
+
+	TEST(bclose(fd) == 0);
+
+	TEST(bsocket_shutdown(NULL) == 0);
+
+	return 0;
+}
+
+int test_listdetach() {
+
+	struct _list *list;
+	struct _list_node *n;
+
+	int i;
+	int elements = 500;
+	int tests = 500;
+
+	int remove_this_guy;
+	int r;
+	int num;
+
+	while (tests--) {
+
+		list = list_new();
+		TEST(list != NULL);
+
+		remove_this_guy = ran(elements);
+
+		n = NULL;
+
+		for (i=0; i<elements; i++) {
+			if (i == remove_this_guy) {
+				n = list_enqueue((void*) i,list);
+				TEST(n != NULL);
+			} else {
+				TEST(list_enqueue((void*) i,list) != NULL);
+			}
+
+		}
+
+		TEST(n != NULL);
+
+		list_node_detach(n);
+
+		for (i=0; i<elements; i++) {
+
+			if (i != remove_this_guy) {
+
+				r = list_dequeue(&num,list);
+				TEST(r == 0);
+
+				TEST(num == i);
+
+			}
+
+		}
+
+		r = list_dequeue(&num,list);
+
+		TEST(r != 0);
+
+		list_free(list);
+
+	}
+
+	return 0;
+}
+
+int test_waitclose_helper(struct _wait_list *wl) {
+
+	int r;
+
+	r = wait_one(wl);
+
+	if (r == -1) {
+		return errno;
+	}
+
+	return 0;
+}
+
+int test_waitclose() {
+
+	HANDLE h;
+
+	struct _wait_list *wl;
+	int r;
+
+	wl = wl_new();
+
+	TEST(wl != NULL);
+
+	h = new_thread(test_waitclose_helper,wl);
+
+	Sleep(100);
+
+	wl_free(wl);
+
+	TEST(WaitForSingleObject(h,INFINITE) == WAIT_OBJECT_0);
+
+	TEST(GetExitCodeThread(h,(DWORD*) &r));
+
+	TEST(r == ECLOSED);
+
+	return 0;
+}
+
+int test_waitfail_helper(struct _wait_list *wl) {
+
+	int r;
+
+	r = wait_one(wl);
+
+	SILENT_TEST(r == 0);
+
+	return 0;
+}
+
+//todo -- what are other situations when this should fail?
+int test_waitfail() {
+
+	HANDLE h;
+
+	struct _wait_list *wl;
+
+	int tests = 10;
+	int r;
+
+	wl = wl_new();
+
+	while (tests--) {
+
+		TEST(wl != NULL);
+
+		h = new_thread(test_waitfail_helper, wl);
+
+		TEST(h != NULL);
+
+		Sleep(10);
+
+		r = wait_one(wl);
+
+		TEST(r == -1);
+		TEST(errno == EAGAIN);
+
+		wl_activate(wl,0);
+
+		TEST(WaitForSingleObject(h,INFINITE) == WAIT_OBJECT_0);
+		TEST(GetExitCodeThread(h,(DWORD*) &r));
+		TEST(r == 0);
+
+		wl_deactivate(wl);
+
+		CloseHandle(h);
+
+	}
+
+	wl_free(wl);
+
+	return 0;
+
+}
+
+int test_waitobject_helper(void *d) {
+
+	struct _wait_list *l;
+	//todo figure out a better way to do this test
+
+
+	l = (struct _wait_list*) d;
+
+	wl_activate((struct _wait_list*) l,0);
+
+	return 0;
+}
+
+//todo think of a more comprehensive test
+int test_waitobject() {
+
+	HANDLE h;
+
+	const int items = 1000;
+
+	struct _wait_list *wl[items];
+	struct _wait_list *desired[2];
+
+	int i;
+	int n,m;
+	int r;
+
+	int tests = 50;
+
+	int wake_this_guy;
+
+	for (i=0; i<items; i++) {
+		wl[i] = wl_new();
+		TEST(wl[i] != NULL);
+	}
+
+	while (tests--) {
+
+		n = ran(items);
+
+		while ( (m = ran(items)) == n);
+
+		TEST(m != n);
+
+		desired[0] = wl[n];
+		desired[1] = wl[m];
+
+		wake_this_guy = ran(2);
+
+		h = new_thread(test_waitobject_helper,desired[wake_this_guy]);
+
+		TEST(WaitForSingleObject(h,INFINITE) == WAIT_OBJECT_0);
+
+		CloseHandle(h);
+
+		r = wait_many(desired,2);
+
+
+	//	printf("%d %d\n",r, wake_this_guy);
+
+		TEST(r == wake_this_guy);
+
+
+		wl_deactivate(desired[wake_this_guy]);
+
+
+	}
+
+	for (i=0; i<items; i++) {
+		wl_free(wl[i]);
+	}
+
+
+	return 0;
+
+}
+
+int simplewait_tests = 100000;
+
+int test_simplewaitobject_helper(struct _wait_list *l) {
+
+	int tests = simplewait_tests;
+
+	while(tests--) {
+		wl_activate(l,0);
+	}
+
+	return 0;
+
+}
+
+int test_simplewaitobject() {
+
+	HANDLE h;
+
+	struct _wait_list *l;
+	int r;
+
+	int tests = simplewait_tests;
+
+	l = wl_new();
+
+	h = new_thread(test_simplewaitobject_helper,l);
+
+	TEST(WaitForSingleObject(h,INFINITE) == WAIT_OBJECT_0);
+
+	CloseHandle(h);
+
+	while(tests--) {
+
+		TEST(l != NULL);
+
+		r = wait_one(l);
+
+		TEST(r == 0);
+
+	}
+
+	/*after this, make sure it blocks*/
+
+	wl_free(l);
+
+	return 0;
+}
+
+
+int test_errnosafe_helper() {
+
+	ASSERT(errno == 0);
+
+	errno = 101;
+	ASSERT( errno == 101);
+
+	return 500;
+
+}
+
+int test_errnosafe() {
+
+	HANDLE h;
+	errno = 102;
+
+	ASSERT(errno == 102);
+
+	h = new_thread(test_errnosafe_helper, NULL);
+
+	TEST(h != NULL);
+	TEST(WaitForSingleObject(h,INFINITE) == WAIT_OBJECT_0);
+
+	CloseHandle(h);
+
+	TEST(errno == 102);
+
+	return 0;
+
+}
+
+int test_maxoverlapped_server(SOCKET s) {
+
+	while ( accept(s,NULL,0) != SOCKET_ERROR);
+
+	return 0;
+}
+
+void CALLBACK test_maxoverlapped_callback (
+ IN DWORD dwError, IN DWORD cbTransferred,
+ IN LPWSAOVERLAPPED lpOverlapped,
+ IN DWORD dwFlags
+) {
+
+
+}
+
+
+int test_maxoverlapped() {
+
+	const int msg_len = 4096;
+
+	HANDLE h;
+
+	SOCKET s;
+	SOCKET c[MAX_BSOCKETS];
+
+	WSABUF buf[MAX_BSOCKETS];
+	WSAOVERLAPPED wo[MAX_BSOCKETS];
+
+	struct sockaddr_in in;
+	struct sockaddr_in name;
+
+	int r;
+	int i;
+
+	int recv, flags;
+
+	s = socket(AF_INET,SOCK_STREAM,0);
+
+	TEST(s != INVALID_SOCKET);
+
+	in.sin_family = AF_INET;
+	in.sin_addr.s_addr = htonl(INADDR_LOOPBACK);
+	in.sin_port = htons(0);
+
+	r = bind(s,(struct sockaddr*) &in,sizeof(struct sockaddr_in));
+	TEST(r != SOCKET_ERROR);
+
+	i = sizeof(struct sockaddr_in);
+	r = getsockname(s,(struct sockaddr*) &name,&i);
+	TEST(r == 0);
+
+	r = listen(s,10);
+	TEST(r != SOCKET_ERROR);
+
+	h = new_thread(test_maxoverlapped_server,(void*) s);
+	TEST(h != NULL);
+
+	for (i=0; i<MAX_BSOCKETS; i++) {
+
+		c[i] = WSASocket(AF_INET,SOCK_STREAM,0,NULL,0,WSA_FLAG_OVERLAPPED);
+		TEST(c[i] != INVALID_SOCKET);
+
+		r  = connect(c[i],(struct sockaddr*) &name,sizeof(struct sockaddr_in));
+		TEST(r != SOCKET_ERROR);
+
+		flags = 0;
+
+		buf[i].buf = malloc(msg_len);
+		buf[i].len = msg_len;
+
+		SetLastError(0);
+
+		r = WSARecv(c[i],&buf[i],1,(DWORD*) &recv, (DWORD*)&flags,&wo[i],test_maxoverlapped_callback);
+
+		TEST(r == SOCKET_ERROR);
+		TEST(WSAGetLastError() == WSA_IO_PENDING);
+	}
+
+	for (i=0; i<MAX_BSOCKETS; i++) {
+		TEST(closesocket(c[i]) == 0);
+	}
+
+	closesocket(s);
+	return 0;
+}
+
+//todo -- come up with more comprehensive test
+int test_bselect_macros() {
+
+	bfd_set fd;
+
+	BFD_ZERO(&fd);
+
+	TEST(fd.count == 0);
+//	TEST(fd.hash[0] == -1);
+
+	BFD_SET(0,&fd);
+
+	TEST(fd.count == 1);
+	TEST(fd.list[0] == 0);
+
+	TEST(fd.hash[0] == 0);
+
+	BFD_SET(1000,&fd);
+
+	TEST(fd.count == 2);
+	TEST(fd.list[0] == 0);
+	TEST(fd.list[1] == 1000);
+
+	TEST(fd.hash[0]  == 0);
+	TEST(fd.hash[1000] == 1);
+
+	BFD_SET(2000,&fd);
+	TEST(BFD_ISSET(0,&fd) );
+	TEST(BFD_ISSET(1000,&fd));
+	TEST(BFD_ISSET(2000,&fd));
+
+	TEST(fd.count == 3);
+	TEST(fd.list[0] == 0);
+	TEST(fd.list[1] == 1000);
+	TEST(fd.list[2] == 2000);
+
+	TEST(fd.hash[0]	   == 0);
+	TEST(fd.hash[1000] == 1);
+	TEST(fd.hash[2000] == 2);
+
+	BFD_CLR(0,&fd);
+	TEST(!BFD_ISSET(0,&fd));
+	TEST(BFD_ISSET(1000,&fd));
+	TEST(BFD_ISSET(2000,&fd));
+
+	TEST(fd.count == 2);
+	TEST(fd.list[0] == 1000);
+	TEST(fd.list[1] == 2000);
+
+
+	//TEST(fd.hash[0] == -1);
+	TEST(fd.hash[1000]	== 0);
+	TEST(fd.hash[2000]  == 1);
+
+	BFD_SET(0,&fd);
+
+	TEST(fd.count == 3);
+	TEST(fd.list[0] == 1000);
+	TEST(fd.list[1] == 2000);
+	TEST(fd.list[2] == 0);
+
+	TEST(fd.hash[0]    == 2);
+	TEST(fd.hash[1000] == 0);
+	TEST(fd.hash[2000] == 1);
+
+	BFD_CLR(2000,&fd);
+	TEST(BFD_ISSET(0,&fd));
+	TEST(BFD_ISSET(1000,&fd));
+	TEST(!BFD_ISSET(2000,&fd));
+
+	TEST(fd.count == 2);
+	TEST(fd.list[0] 	== 1000);
+	TEST(fd.list[1]		== 0);
+
+	TEST(fd.hash[0]		== 1);
+	TEST(fd.hash[1000]	== 0);
+//	TEST(fd.hash[2000] 	== -1);
+
+	BFD_SET(2000,&fd);
+
+	TEST(fd.count == 3);
+	TEST(fd.list[0] == 1000);
+	TEST(fd.list[1] == 0);
+	TEST(fd.list[2] == 2000);
+
+	TEST(fd.hash[0] == 1);
+	TEST(fd.hash[1000] == 0);
+	TEST(fd.hash[2000] == 2);
+
+	BFD_CLR(2000,&fd);
+	TEST(BFD_ISSET(0,&fd));
+	TEST(BFD_ISSET(1000,&fd));
+	TEST(!BFD_ISSET(2000,&fd));
+
+	TEST(fd.count == 2);
+	TEST(fd.list[0] == 1000);
+	TEST(fd.list[1]	== 0);
+
+	TEST(fd.hash[0]		== 1);
+	TEST(fd.hash[1000]	== 0);
+	//TEST(fd.hash[2000]  == -1);
+
+	BFD_ZERO(&fd);
+
+	TEST(fd.count == 0);
+	TEST(!BFD_ISSET(0,&fd));
+	TEST(!BFD_ISSET(1000,&fd));
+	TEST(!BFD_ISSET(2000,&fd));
+
+//	TEST(fd.hash[0] == -1);
+//	TEST(fd.hash[1000] == -1);
+//	TEST(fd.hash[2000] == -1);
+
+	return 0;
+}
+
+int test_connect_google( ) {
+
+	int fd;
+	int r;
+
+	TEST(bsocket_init(NULL) == 0);
+
+	fd = bsocket(AF_INET,SOCK_STREAM,0);
+
+	TEST(fd == 0);
+
+	google.sin_port = htons(80);
+
+	r = bconnect(fd,(struct sockaddr*) &google,sizeof(google));
+
+	TEST(r == 0);
+
+	r = bclose(fd);
+
+	TEST(r == 0);
+
+	TEST(bsocket_shutdown(NULL) == 0);
+
+	return 0;
+
+}
+
+int test_setsocketerr() {
+
+	int fd;
+
+	int err;
+
+	TEST(bsocket_init(NULL) == 0);
+
+	fd = bsocket(AF_INET,SOCK_STREAM,0);
+
+	TEST(fd == 0);
+
+	socket_set_err(fd,666,__GLOBAL_BSOCKET_ENV_);
+
+	socket_last_err(fd,&err,__GLOBAL_BSOCKET_ENV_);
+
+	TEST(err == 666);
+
+	TEST(bsocket_shutdown(NULL) == 0);
+
+	return 0;
+
+}
+
+int test_multi_connect()  {
+
+	const int clients = 300;
+
+	bfd_set fds_out, fds_in;
+	struct sockaddr_in in;
+
+	int err;
+	int len;
+	int fd[clients];
+	int i;
+	int s;
+
+	int r;
+	int z;
+
+	int port;
+	int counter;
+	int good;
+
+	s = loopback_server(&port);
+
+	TEST(s != 0);
+
+	TEST(bsocket_init(NULL) == 0);
+
+	in.sin_family = AF_INET;
+	in.sin_addr.s_addr = htonl(INADDR_LOOPBACK);
+	in.sin_port = port;
+
+	BFD_ZERO(&fds_in);
+
+	for (i=0; i<clients; i++) {
+
+		fd[i] = bsocket(AF_INET,SOCK_STREAM,0);
+
+		TEST(fd[i] >= 0);
+
+		TEST(bfcntl(fd[i],F_SETFL,O_NONBLOCK) == 0);
+
+		TEST(fd[i] != -1);
+		BFD_SET(fd[i],&fds_in);
+	}
+
+	for (i=0; i<clients; i++) {
+		r = bconnect(fd[i],(struct sockaddr*) &in, sizeof(in));
+
+		TEST(r == -1);
+		TEST(errno == EINPROGRESS);
+	}
+
+	BFD_COPY(&fds_out,&fds_in);
+
+	counter = 0;
+	good = FALSE;
+	while (
+
+		(r = bselect(0,NULL,&fds_out,NULL,NULL)) == 1) {
+
+		len = sizeof(int);
+
+		z = bgetsockopt( fds_out.list[0],SOL_SOCKET,SO_ERROR,&err,&len);
+
+		TEST(z == 0);
+
+		if (err == 0) {
+			good = TRUE;
+		}
+
+
+		BFD_CLR(fds_out.list[0], &fds_in);
+
+		BFD_COPY(&fds_out,&fds_in);
+		counter++;
+	}
+
+	//make sure at least one connection succeeded
+	TEST(good);
+
+
+	TEST(counter == clients);
+
+	TEST(r == -1);
+
+	TEST(fds_out.count == 0);
+
+	for (i=0; i<clients; i++) {
+		TEST(bclose(fd[i]) == 0);
+	}
+
+	//todo -- why does this fail
+	//TEST(closesocket(s) == 0);
+	closesocket(s);
+
+
+	TEST(bsocket_shutdown(NULL) == 0);
+
+	return 0;
+}
+
+
+int test_simple_write_helper() {
+
+	SOCKET s;
+	SOCKET c;
+
+	int r;
+
+	s = socket(AF_INET,SOCK_STREAM,0);
+
+	SILENT_TEST(s != INVALID_SOCKET);
+
+	SILENT_TEST(bind(s,(struct sockaddr*) &localhost,sizeof(localhost)) == 0);
+
+	SILENT_TEST(listen(s,1) == 0);
+
+	c = accept(s,NULL,0);
+
+	SILENT_TEST(c != INVALID_SOCKET);
+
+	r = -1;
+
+	SILENT_TEST(recv(c,(char*) &r,sizeof(r),0) == sizeof(r));
+
+	SILENT_TEST(r == 666);
+
+	closesocket(s);
+	closesocket(c);
+
+	return 0;
+
+}
+
+int test_simple_write() {
+
+	HANDLE h;
+
+	int msg;
+
+	int fd;
+	int r;
+
+	TEST(bsocket_init(NULL) == 0);
+
+	/*start server*/
+
+	h = new_thread(test_simple_write_helper,NULL);
+	Sleep(100);
+
+	fd = bsocket(AF_INET,SOCK_STREAM,0);
+
+	TEST(fd == 0);
+
+	r = bconnect(fd, (struct sockaddr *) &localhost,sizeof(localhost));
+
+	TEST(r == 0);
+
+	msg = 666;
+
+	r = bsend(fd,&msg,sizeof(msg),0);
+
+	TEST(r == sizeof(msg));
+
+	TEST(WaitForSingleObject(h,1000) == WAIT_OBJECT_0);
+
+	TEST(GetExitCodeThread(h,(DWORD*) &r));
+
+	TEST(r == 0);
+
+	TEST(bsocket_shutdown(NULL) == 0);
+
+	return 0;
+}
+
+int test_simple_read_helper() {
+
+	SOCKET s;
+	SOCKET c;
+
+	int r;
+	int msg;
+
+
+	s = socket(AF_INET,SOCK_STREAM,0);
+
+	SILENT_TEST(s != INVALID_SOCKET);
+
+	r = bind(s,(struct sockaddr*) &localhost,sizeof(localhost));
+	SILENT_TEST(r == 0);
+
+	r = listen(s,10);
+	SILENT_TEST(r == 0);
+
+	c = accept(s,NULL,0);
+	SILENT_TEST(c != INVALID_SOCKET);
+
+	msg = 666;
+	Sleep(100);
+
+	r = send(c,(char*) &msg, sizeof(msg),0);
+	SILENT_TEST(r == sizeof(msg));
+
+	closesocket(s);
+	closesocket(c);
+
+	return 0;
+
+}
+
+int test_simple_read() {
+
+	HANDLE h;
+
+	int fd;
+	int r;
+	int msg;
+
+	TEST(bsocket_init(NULL) == 0);
+
+	h = new_thread(test_simple_read_helper,NULL);
+	TEST(h != NULL);
+
+	fd = bsocket(AF_INET,SOCK_STREAM,0);
+	TEST(fd == 0);
+
+	r = bconnect(fd,(struct sockaddr*) &localhost,sizeof(localhost));
+	TEST(r == 0);
+
+	r = brecv(fd,&msg,sizeof(msg),0);
+
+	TEST(r == sizeof(msg));
+	TEST(msg == 666);
+
+	TEST(bsocket_shutdown(NULL) == 0);
+
+	return 0;
+
+}
+
+int test_verify_npp_usage() {
+	errno = 666;
+	return -1;
+}
+
+
+int test_complex_select() {
+	return -1;
+}
+
+
+struct test_case tc[] =
+
+{
+	{test_errnosafe,NULL,"Check if errno is thread safe."},
+	{test_createlist,NULL,"Create and destroy list"},
+	{test_basicqueue,NULL,"Create list, add and remove single element. Check integrity."},
+	{test_bigqueue,NULL,"Insertion and removal of a large number of random elements into queue."},
+	{test_listdetach,NULL,"Test detaching node from list."},
+	{test_envmutex,NULL,"Test environment mutexes."},
+	{test_envmutexclose,NULL,"Test if shutdown properly closes mutexes."},
+	{test_setsocketerr,NULL,"Test setting socket error."},
+	{test_socketmutex,NULL,"Test socket mutexes."},
+	{test_socketmutexclose,NULL,"Test if socket close properly releases mutex ownership."},
+	{test_syncstruct,NULL,"Simple usage of synchronization objects."},
+	{test_badsocket,NULL,"Test is mutex fails for non-existent socket."},
+	{test_socketinit,NULL,"Test indempotency of init operation"},
+	{test_overflowsocket,NULL,"Testing if bsocket() fails after creating too many sockets."},
+	{test_simplewaitobject,NULL,"Testing simple usage of wait objects."},
+	{test_waitobject,NULL,"Test more complicated usage of wait objects."},
+	{test_waitfail,NULL,"Tests if a wait object fails when it should."},
+	{test_waitclose,NULL,"Test if a wait object behaves properly on closure."},
+	{test_maxoverlapped,NULL,"Test if we can have a sufficient number of outstanding operations."},
+	{test_bselect_macros,NULL,"Testing bselect() macros."},
+//	{test_multieventselect,NULL,"Can we have more than one socket associated with a WSAevent."},
+	//{test_theory1,NULL,"Test theory #1 (WSAwfme does not block when socket closed."},
+	//todo -- find a way to reduce timeout
+	//{test_badconnection_blocking,NULL,"Test if connection to unreachable host fails appropriately. (blocking)"},
+	{test_parallel_connect,NULL,"Test if multiple connections to same socket fail appropriately."},
+	{test_connect_close,NULL,"Test if close() while waiting on connect causes problems."},
+	{test_connect_google,NULL,"Test connection to google.com"},
+	{test_active_refusal_blocking,NULL,"Test if active refusal handled properly. (blocking)"},
+	{test_active_refusal_nonblocking,NULL,"\t(non-blocking)"},
+	{test_multi_connect,NULL,"Test rapid firing server with connections."},
+	{test_simple_write,NULL,"Connect and send some data to a server."},
+	{test_simple_read,NULL,"Connect and read some data from a server."},
+	{test_verify_npp_usage,NULL,"Check that the NPP is not being used for large sends."},
+	{NULL,NULL,NULL}
+};
+
+//todo -- one of the tests is opening and not closing a large number of handles
+int main(int argc, char **argv) {
+
+	struct hostent *host;
+
+	/*todo warn user about this putting strain on system*/
+	int i = 0;
+
+	int pass = 0;
+	int fail = 0;
+
+	srand(time(NULL));
+
+	//todo uncomment
+	winsock_start();
+
+	//lookup googles address before starting (we want something that should be available)
+
+	host = gethostbyname("google.com");
+	ASSERT(host != NULL);
+
+	google.sin_family = AF_INET;
+	google.sin_addr = *(struct in_addr *) host->h_addr;
+
+	localhost.sin_family = AF_INET;
+	localhost.sin_addr.s_addr = inet_addr("127.0.0.1");
+	localhost.sin_port = htons(81);
+
+	if (argc > 1) {
+		i = atoi(argv[1]);
+	}
+
+	while (tc[i].fun != NULL) {
+
+		printf("Test #%d: %s\n",i,tc[i].desc);
+		fflush(stdout);
+
+		if ( tc[i].fun() ) {
+
+			printf("\n");
+			fflush(stdout);
+			fail++;
+		} else {
+			pass++;
+		}
+
+		i++;
+
+		if (argc > 1)
+			break;
+
+	}
+
+	printf("\nTesting Report: %d passed, %d failed\n",pass,fail);
+
+	return 0;
+
+}
+


Property changes on: bsockets/trunk/test.c
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/test.h
===================================================================
--- bsockets/trunk/test.h	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/test.h	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,10 @@
+#ifndef _TEST_H_
+#define _TEST_H
+
+#define TEST(X)  			if (!((res = X))) { printf("Assertion failed: \"%s\" [errno:%d, w32err:%d line:%d]\n", #X,errno,(int) GetLastError(),__LINE__); fflush(stdout); return 1; }
+#define SILENT_TEST(X)		if (!(X)) { return GetLastError(); }
+
+#endif
+
+
+


Property changes on: bsockets/trunk/test.h
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/todo
===================================================================
--- bsockets/trunk/todo	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/todo	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,10 @@
+todo
+-----
+
+- make getsockopt()
+- make listen()
+- make accept()
+- make read()
+- make write()
+- make socketpair()
+- make bsocket.c (links to system specific versions)


Property changes on: bsockets/trunk/todo
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/unix.c
===================================================================
--- bsockets/trunk/unix.c	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/unix.c	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,49 @@
+#include <windows.h>
+
+#include "unix.h"
+
+
+int unixify_wsaerr(DWORD err) {
+
+	int out = -1;
+
+	switch (err) {
+
+		case 0:
+			out = 0;
+		break;
+
+		case WSANOTINITIALISED:
+			out = ENOTINIT;
+		break;
+
+		case WSAEMFILE:
+			out = EMFILE;
+		break;
+
+		case WSAENOBUFS:
+			out = ENOBUFS;
+		break;
+
+		case WSAETIMEDOUT:
+			out = ETIMEDOUT;
+		break;
+
+		case WSAEWOULDBLOCK:
+			out = EINPROGRESS;
+		break;
+
+		case WSAECONNREFUSED:
+			out = ECONNREFUSED;
+		break;
+
+		default:
+			out = err;
+		break;
+
+	}
+
+	return out;
+
+}
+


Property changes on: bsockets/trunk/unix.c
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/unix.h
===================================================================
--- bsockets/trunk/unix.h	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/unix.h	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,68 @@
+#ifndef _UNIX_H_
+#define _UNIX_H_
+
+#include <windows.h>
+#include <errno.h>
+
+int unixify_wsaerr(DWORD);
+
+#define BSOCK_PRIVATE_ERRNO 110000
+
+//todo check that these are in private ranges
+#ifndef ENOBUFS
+#define ENOBUFS			(BSOCK_PRIVATE_ERRNO+1)
+#endif
+
+#ifndef EMFILE
+#define EMFILE			(BSOCK_PRIVATE_ERRNO+2)
+#endif
+
+#ifndef ENOSYS
+#define ENOSYS			(BSOCK_PRIVATE_ERRNO+3)
+#endif
+
+#ifndef EINVAL
+#define EINVAL			(BSOCK_PRIVATE_ERRNO+4)
+#endif
+
+#ifndef ENOTINIT
+#define ENOTINIT		(BSOCK_PRIVATE_ERRNO+5)
+#endif
+
+#ifndef EALREADY
+#define EALREADY		(BSOCK_PRIVATE_ERRNO+6)
+#endif
+
+#ifndef EISCONN
+#define EISCONN			(BSOCK_PRIVATE_ERRNO+7)
+#endif
+
+#ifndef EBADF
+#define EBADF			(BSOCK_PRIVATE_ERRNO+8)
+#endif
+
+#ifndef EINPROGRESS
+#define	EINPROGRESS		(BSOCK_PRIVATE_ERRNO+9)
+#endif
+
+#ifndef ETIMEDOUT
+#define	ETIMEDOUT		(BSOCK_PRIVATE_ERRNO+10)
+#endif
+
+#ifndef ECONNREFUSED
+#define ECONNREFUSED	(BSOCK_PRIVATE_ERRNO+11)
+#endif
+
+#ifndef EEMPTY
+#define EEMPTY			(BSOCK_PRIVATE_ERRNO+12)
+#endif
+
+#ifndef ENOTSUP
+#define ENOTSUP			(BSOCK_PRIVATE_ERRNO+13)
+#endif
+
+#ifndef ECLOSED
+#define ECLOSED			(BSOCK_PRIVATE_ERRNO+14)
+#endif
+
+#endif


Property changes on: bsockets/trunk/unix.h
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/wait.c
===================================================================
--- bsockets/trunk/wait.c	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/wait.c	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,346 @@
+#include <windows.h>
+
+#include "unix.h"
+#include "list.h"
+#include "wait.h"
+#include "misc.h"
+
+
+//todo, come up with a way to recycle monitor objects so we dont have
+//keep allocating them for each wait operation
+
+//todo -- resolve external race conditions with AE_LIST
+
+//todo -- when activated with error code, each wait should return with that error code
+int monitor_link(struct _wait_list *l, struct _monitor *m) {
+
+	int out;
+
+	out = 0;
+
+	ASSERT(l != NULL);
+	ASSERT(m != NULL);
+
+
+	if (l->active) {
+
+		out = 1;
+
+	} else {
+
+		if (l->waiting == NULL) {
+
+			CHECK(list_enqueue(l, m->where) != NULL,0);
+
+			l->waiting = m;
+
+
+		} else {
+
+			errno = EAGAIN;
+			out = -1;
+
+		}
+
+	}
+
+	fail:
+
+	return out;
+}
+
+void monitor_detach(struct _monitor *m) {
+
+	struct _wait_list *l;
+
+	ASSERT(m != NULL);
+
+	while (list_dequeue(&l,m->where) == 0) {
+
+		ASSERT(l != NULL);
+		ASSERT(l->waiting == m);
+
+		l->waiting = NULL;
+
+	}
+
+	ASSERT(m->where->head == NULL);
+	ASSERT(m->where->tail == NULL);
+
+}
+
+//todo -- this function should not require any time of atomicity
+void monitor_wait(struct _monitor *m, int timeout) {
+
+	int r;
+
+	ASSERT(m != NULL);
+
+	if (timeout == -1) {
+		timeout = INFINITE;
+	}
+
+
+	r = WaitForSingleObject(m->event,timeout);
+
+	switch(r) {
+
+		case WAIT_OBJECT_0:
+			ASSERT(
+				(m->who != NULL) ||
+				(m->who == NULL && m->err != 0)
+				);
+
+		break;
+
+
+		case WAIT_TIMEOUT:
+			ASSERT(timeout != INFINITE);
+			m->who = NULL;
+			m->err = ETIMEDOUT;
+			monitor_detach(m);
+		break;
+
+
+		default:
+			ASSERT(FALSE);
+		break;
+
+
+	}
+
+	ASSERT(m->where != NULL);
+	ASSERT(m->where->head == NULL);
+
+	if (m->who == NULL) {
+		ASSERT(m->err != 0);
+	}
+
+}
+
+void monitor_free(struct _monitor *m) {
+
+	ASSERT(m != NULL);
+
+	monitor_detach(m);
+
+	if (m->event != NULL)
+		ASSERT(CloseHandle(m->event));
+
+	if (m->where != NULL)
+		list_free(m->where);
+
+	free(m);
+
+}
+
+struct _monitor *monitor_new() {
+
+	struct _monitor *m;
+	int out;
+	int err;
+
+	out = 0;
+	err = 0;
+
+	m = (struct _monitor*)
+		malloc(sizeof(struct _monitor));
+
+	CHECK(m != NULL,ENOMEM);
+
+	m->where = list_new();
+	CHECK(m->where != NULL,0);
+
+	m->event = CreateEvent(NULL,FALSE,FALSE,NULL);
+	CHECK_(m->event != NULL);
+
+	fail:
+	err = errno;
+
+	if (out == -1) {
+
+		if (m != NULL) {
+			ASSERT(FALSE);
+			monitor_free(m);
+			m = NULL;
+		}
+
+	}
+
+	errno = err;
+	return m;
+}
+
+//todo -- get rid of multiple release mutexes
+//this code is a nightmare, clean it up
+int wait_many_timeout(struct _wait_list **d, int n, int timeout) {
+
+	struct _monitor *m;
+
+	ASSERT(d != NULL);
+
+	int i;
+	int out;
+	int r;
+
+
+	m = monitor_new();
+
+	CHECK(m != NULL,0);
+
+
+	for (i=0; i<n; i++) {
+
+		r = monitor_link(d[i],m);
+
+		CHECK(r != -1,0);
+
+		if (r == 1) {
+
+			//we have found one which is open, no need to keep looking
+			monitor_free(m);
+
+			//todo -- i hate this
+			return i;
+
+		} else {
+			ASSERT(r == 0)
+		}
+
+	}
+
+	monitor_wait(m,timeout);
+
+	//todo -- why was this here?
+	//ASSERT(m->who != NULL);
+
+	out = -1;
+	if (m->err) {
+
+		errno = m->err;
+
+		CHECK(FALSE,0);
+	}
+
+	//todo -- there must be a way to do this in constant time
+	for(i=0; i<n; i++) {
+		if (d[i] == m->who) {
+			out = i;
+			i = n+1;
+		}
+	}
+
+	ASSERT(out >= 0);
+	ASSERT(out < n);
+
+	fail:
+
+	if (m != NULL) {
+		monitor_free(m);
+	}
+
+	return out;
+}
+
+int wait_many(struct _wait_list **d, int n) {
+	return wait_many_timeout(d,n,-1);
+}
+
+int wait_one(struct _wait_list *item) {
+
+	struct _wait_list *d[1];
+
+	d[0] = item;
+
+	return wait_many(d,1);
+
+}
+
+/*
+void wl_open(struct _wait_list *wl) {
+
+	ASSERT(wl != NULL);
+
+	wl_activate(wl,0);
+	wl->active = -1;
+
+}
+
+void wl_close(struct _wait_list *wl)  {
+
+	ASSERT(wl != NULL);
+	//you are not allowed to call wl_close on a closed lists
+	ASSERT(wl->waiting == NULL);
+
+	wl->active = 0;
+
+}
+
+*/
+
+void wl_activate(struct _wait_list *l, int code) {
+
+	struct _monitor *m;
+
+	ASSERT(l != NULL);
+
+	if ( (m = l->waiting) != NULL) {
+
+
+		monitor_detach(m);
+		m->who = l;
+		m->err = code;
+
+		SetEvent(m->event);
+
+	}
+
+
+	l->active = TRUE;
+
+
+}
+
+void wl_deactivate(struct _wait_list *l) {
+
+	ASSERT(l->waiting == NULL);
+	ASSERT(l->active);
+
+	l->active = FALSE;
+
+}
+
+void wl_free (struct _wait_list *l){
+
+	wl_activate(l,ECLOSED);
+
+	ASSERT(l != NULL)
+
+	free(l);
+
+}
+
+struct _wait_list *wl_new() {
+
+	struct _wait_list *out;
+
+	out =  (struct _wait_list*)	malloc(sizeof(struct _wait_list));
+
+	if (out != NULL) {
+
+		out->waiting = NULL;
+		out->active = 0;
+
+	} else {
+
+		errno = ENOMEM;
+
+	}
+
+	return out;
+}
+
+
+
+
+


Property changes on: bsockets/trunk/wait.c
___________________________________________________________________
Name: svn:executable
   + 

Added: bsockets/trunk/wait.h
===================================================================
--- bsockets/trunk/wait.h	2006-07-27 03:17:50 UTC (rev 6916)
+++ bsockets/trunk/wait.h	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,53 @@
+#ifndef _WAIT_H_
+#define _WAIT_H_
+
+struct _wait_list {
+
+	/*a queue of monitors who want to be woken up in order when data is
+		available*/
+
+	struct _monitor *waiting;
+
+	int active;
+
+};
+
+
+struct _monitor {
+
+	/*whoever is waiting on this monitor is waiting for this event
+		to be triggered before doing anything*/
+
+	HANDLE event;
+
+	/*nodes which are in lines in which we are currently waiting*/
+	struct _list *where;
+
+
+	/*a pointer to the object who woke us up */
+	struct _wait_list *who;
+
+	int err;
+	int timeout;
+
+};
+
+
+
+struct _wait_list *wl_new();
+void wl_free();
+
+int wait_one(struct _wait_list *);
+int wait_many(struct _wait_list **, int);
+
+int wait_many_timeout(struct _wait_list **, int, int);
+
+void wl_open(struct _wait_list *);
+void wl_close(struct _wait_list *);
+
+void wl_activate(struct _wait_list *, int);
+void wl_deactivate(struct _wait_list *);
+
+//void wl_wakeall(struct _wait_list *, int);
+
+#endif


Property changes on: bsockets/trunk/wait.h
___________________________________________________________________
Name: svn:executable
   + 

Added: svn-commit.tmp
===================================================================
--- svn-commit.tmp	2006-07-27 03:17:50 UTC (rev 6916)
+++ svn-commit.tmp	2006-07-27 03:47:11 UTC (rev 6917)
@@ -0,0 +1,5 @@
+Inital import for Mike Chiussi's bsockets library.
+
+--This line, and those below, will be ignored--
+
+A    .