Cleaned it up so that code paths which have unsupported WS_SO values
never actually pass them to Unix system calls.

diff --git a/dlls/winsock/socket.c b/dlls/winsock/socket.c
index 102a4ba..d4585f8 100644
--- a/dlls/winsock/socket.c
+++ b/dlls/winsock/socket.c
@@ -178,18 +178,23 @@
 static INT         _ws_sock_ops[] =
        { WS_SO_DEBUG, WS_SO_REUSEADDR, WS_SO_KEEPALIVE, WS_SO_DONTROUTE,
          WS_SO_BROADCAST, WS_SO_LINGER, WS_SO_OOBINLINE, WS_SO_SNDBUF,
-         WS_SO_RCVBUF, WS_SO_ERROR, WS_SO_TYPE, WS_SO_DONTLINGER,
+         WS_SO_RCVBUF, WS_SO_ERROR, WS_SO_TYPE,
 #ifdef SO_RCVTIMEO
 	 WS_SO_RCVTIMEO,
 #endif
+#ifdef SO_SNDTIMEO
+	 WS_SO_SNDTIMEO,
+#endif
 	 0 };
 static int           _px_sock_ops[] =
        { SO_DEBUG, SO_REUSEADDR, SO_KEEPALIVE, SO_DONTROUTE, SO_BROADCAST,
          SO_LINGER, SO_OOBINLINE, SO_SNDBUF, SO_RCVBUF, SO_ERROR, SO_TYPE,
-	 WS_SO_DONTLINGER,  /* no unix equivalent */
 #ifdef SO_RCVTIMEO
 	 SO_RCVTIMEO,
 #endif
+#ifdef SO_SNDTIMEO
+	 SO_SNDTIMEO,
+#endif
 	};
 
 static INT _ws_tcp_ops[] = {
@@ -390,8 +395,9 @@
  *          convert_sockopt()
  *
  * Converts socket flags from Windows format.
+ * Return 1 if converted, 0 if not (error).
  */
-static void convert_sockopt(INT *level, INT *optname)
+static int convert_sockopt(INT *level, INT *optname)
 {
   int i;
   switch (*level)
@@ -400,17 +406,24 @@
         *level = SOL_SOCKET;
         for(i=0; _ws_sock_ops[i]; i++)
             if( _ws_sock_ops[i] == *optname ) break;
-        if( _ws_sock_ops[i] ) *optname = _px_sock_ops[i];
-        else FIXME("Unknown SOL_SOCKET optname %d\n", *optname);
+        if( _ws_sock_ops[i] ) {
+	    *optname = _px_sock_ops[i];
+	    return 1;
+	}
+        FIXME("Unknown SOL_SOCKET optname 0x%x\n", *optname);
         break;
      case WS_IPPROTO_TCP:
         *level = IPPROTO_TCP;
         for(i=0; _ws_tcp_ops[i]; i++)
 		if ( _ws_tcp_ops[i] == *optname ) break;
-        if( _ws_tcp_ops[i] ) *optname = _px_tcp_ops[i];
-        else FIXME("Unknown IPPROTO_TCP optname %d\n", *optname);
+        if( _ws_tcp_ops[i] ) {
+	    *optname = _px_tcp_ops[i];
+	    return 1;
+	}
+        FIXME("Unknown IPPROTO_TCP optname 0x%x\n", *optname);
 	break;
   }
+  return 0;
 }
 
 /* ----------------------------------- Per-thread info (or per-process?) */
@@ -1224,13 +1237,16 @@
     if( _check_ws(pwsi, s) )
     {
 	int fd = _get_sock_fd(s);
-	convert_sockopt(&level, &optname);
-	if (getsockopt(fd, (int) level, optname, optval, optlen) == 0 )
-	{
-	    close(fd);
-	    return 0;
+	if (!convert_sockopt(&level, &optname)) {
+	    SetLastError(WSAENOPROTOOPT);	/* Unknown option */
+        } else {
+	    if (getsockopt(fd, (int) level, optname, optval, optlen) == 0 )
+	    {
+		close(fd);
+		return 0;
+	    }
+	    SetLastError((errno == EBADF) ? WSAENOTSOCK : wsaErrno());
 	}
-	SetLastError((errno == EBADF) ? WSAENOTSOCK : wsaErrno());
 	close(fd);
     }
     return SOCKET_ERROR;
@@ -2064,28 +2080,33 @@
 	int fd = _get_sock_fd(s);
         int woptval;
 
-        convert_sockopt(&level, &optname);
         if(optname == WS_SO_DONTLINGER) {
+	    /* This is unique to WinSock and takes special conversion */
             linger.l_onoff	= *((int*)optval) ? 0: 1;
             linger.l_linger	= 0;
             optname=SO_LINGER;
             optval = (char*)&linger;
             optlen = sizeof(struct linger);
         }else{
-            if (optname == SO_LINGER && optval) {
-		/* yes, uses unsigned short in both win16/win32 */
-		linger.l_onoff	= ((UINT16*)optval)[0];
-		linger.l_linger	= ((UINT16*)optval)[1];
-		/* FIXME: what is documented behavior if SO_LINGER optval
-		   is null?? */
-		optval = (char*)&linger;
-		optlen = sizeof(struct linger);
-            } else if (optlen < sizeof(int)){
-                woptval= *((INT16 *) optval);
-                optval= (char*) &woptval;
-                optlen=sizeof(int);
-            }
-        }
+            if (!convert_sockopt(&level, &optname)) {
+		SetLastError(WSAENOPROTOOPT);
+		close(fd);
+		return SOCKET_ERROR;
+	    }
+	}
+	if (optname == SO_LINGER && optval) {
+	    /* yes, uses unsigned short in both win16/win32 */
+	    linger.l_onoff	= ((UINT16*)optval)[0];
+	    linger.l_linger	= ((UINT16*)optval)[1];
+	    /* FIXME: what is documented behavior if SO_LINGER optval
+	       is null?? */
+	    optval = (char*)&linger;
+	    optlen = sizeof(struct linger);
+	} else if (optlen < sizeof(int)){
+	    woptval= *((INT16 *) optval);
+	    optval= (char*) &woptval;
+	    optlen=sizeof(int);
+	}
 	if (setsockopt(fd, level, optname, optval, optlen) == 0)
 	{
 	    close(fd);