Commit 1b66fb03 authored by Pavel Shilovsky's avatar Pavel Shilovsky

Fix tunnel port problem for 2.6.33

parent 698c4776
...@@ -1391,6 +1391,10 @@ cifs_find_tcp_session(struct sockaddr_storage *addr, unsigned short int port) ...@@ -1391,6 +1391,10 @@ cifs_find_tcp_session(struct sockaddr_storage *addr, unsigned short int port)
struct sockaddr_in *addr4 = (struct sockaddr_in *) addr; struct sockaddr_in *addr4 = (struct sockaddr_in *) addr;
struct sockaddr_in6 *addr6 = (struct sockaddr_in6 *) addr; struct sockaddr_in6 *addr6 = (struct sockaddr_in6 *) addr;
/* looking for CIFS_PORT if user doens't specify the port manually */
if (!port)
port = CIFS_PORT;
write_lock(&cifs_tcp_ses_lock); write_lock(&cifs_tcp_ses_lock);
list_for_each(tmp, &cifs_tcp_ses_list) { list_for_each(tmp, &cifs_tcp_ses_list) {
server = list_entry(tmp, struct TCP_Server_Info, server = list_entry(tmp, struct TCP_Server_Info,
...@@ -1406,34 +1410,23 @@ cifs_find_tcp_session(struct sockaddr_storage *addr, unsigned short int port) ...@@ -1406,34 +1410,23 @@ cifs_find_tcp_session(struct sockaddr_storage *addr, unsigned short int port)
switch (addr->ss_family) { switch (addr->ss_family) {
case AF_INET: case AF_INET:
if (addr4->sin_addr.s_addr ==
server->addr.sockAddr.sin_addr.s_addr) {
addr4->sin_port = htons(port); addr4->sin_port = htons(port);
/* user overrode default port? */ if ((addr4->sin_addr.s_addr !=
if (addr4->sin_port) { server->addr.sockAddr.sin_addr.s_addr) ||
if (addr4->sin_port != (addr4->sin_port != server->addr.sockAddr.sin_port))
server->addr.sockAddr.sin_port)
continue; continue;
}
break; break;
} else
continue;
case AF_INET6: case AF_INET6:
if (ipv6_addr_equal(&addr6->sin6_addr,
&server->addr.sockAddr6.sin6_addr) &&
(addr6->sin6_scope_id ==
server->addr.sockAddr6.sin6_scope_id)) {
addr6->sin6_port = htons(port); addr6->sin6_port = htons(port);
/* user overrode default port? */ if (!ipv6_addr_equal(&addr6->sin6_addr,
if (addr6->sin6_port) { &server->addr.sockAddr6.sin6_addr) ||
if (addr6->sin6_port != (addr6->sin6_scope_id !=
server->addr.sockAddr6.sin6_port) server->addr.sockAddr6.sin6_scope_id) ||
(addr6->sin6_port !=
server->addr.sockAddr6.sin6_port))
continue; continue;
}
break; break;
} else
continue;
} }
++server->srv_count; ++server->srv_count;
...@@ -1774,7 +1767,7 @@ ipv4_connect(struct TCP_Server_Info *server) ...@@ -1774,7 +1767,7 @@ ipv4_connect(struct TCP_Server_Info *server)
int rc = 0; int rc = 0;
int val; int val;
bool connected = false; bool connected = false;
__be16 orig_port = 0; bool orig_port_error = false;
struct socket *socket = server->ssocket; struct socket *socket = server->ssocket;
if (socket == NULL) { if (socket == NULL) {
...@@ -1792,20 +1785,18 @@ ipv4_connect(struct TCP_Server_Info *server) ...@@ -1792,20 +1785,18 @@ ipv4_connect(struct TCP_Server_Info *server)
cifs_reclassify_socket4(socket); cifs_reclassify_socket4(socket);
} }
/* user overrode default port */ /* user overrode default port or we perform reconnect */
if (server->addr.sockAddr.sin_port) { if (server->addr.sockAddr.sin_port) {
rc = socket->ops->connect(socket, (struct sockaddr *) rc = socket->ops->connect(socket, (struct sockaddr *)
&server->addr.sockAddr, &server->addr.sockAddr,
sizeof(struct sockaddr_in), 0); sizeof(struct sockaddr_in), 0);
if (rc >= 0) if (rc >= 0)
connected = true; connected = true;
else
orig_port_error = true;
} }
if (!connected) { if (!orig_port_error && !connected) {
/* save original port so we can retry user specified port
later if fall back ports fail this time */
orig_port = server->addr.sockAddr.sin_port;
/* do not retry on the same port we just failed on */ /* do not retry on the same port we just failed on */
if (server->addr.sockAddr.sin_port != htons(CIFS_PORT)) { if (server->addr.sockAddr.sin_port != htons(CIFS_PORT)) {
server->addr.sockAddr.sin_port = htons(CIFS_PORT); server->addr.sockAddr.sin_port = htons(CIFS_PORT);
...@@ -1817,7 +1808,8 @@ ipv4_connect(struct TCP_Server_Info *server) ...@@ -1817,7 +1808,8 @@ ipv4_connect(struct TCP_Server_Info *server)
connected = true; connected = true;
} }
} }
if (!connected) {
if (!orig_port_error && !connected) {
server->addr.sockAddr.sin_port = htons(RFC1001_PORT); server->addr.sockAddr.sin_port = htons(RFC1001_PORT);
rc = socket->ops->connect(socket, (struct sockaddr *) rc = socket->ops->connect(socket, (struct sockaddr *)
&server->addr.sockAddr, &server->addr.sockAddr,
...@@ -1829,15 +1821,12 @@ ipv4_connect(struct TCP_Server_Info *server) ...@@ -1829,15 +1821,12 @@ ipv4_connect(struct TCP_Server_Info *server)
/* give up here - unless we want to retry on different /* give up here - unless we want to retry on different
protocol families some day */ protocol families some day */
if (!connected) { if (!connected) {
if (orig_port)
server->addr.sockAddr.sin_port = orig_port;
cFYI(1, ("Error %d connecting to server via ipv4", rc)); cFYI(1, ("Error %d connecting to server via ipv4", rc));
sock_release(socket); sock_release(socket);
server->ssocket = NULL; server->ssocket = NULL;
return rc; return rc;
} }
/* /*
* Eventually check for other socket options to change from * Eventually check for other socket options to change from
* the default. sock_setsockopt not used because it expects * the default. sock_setsockopt not used because it expects
...@@ -1935,7 +1924,7 @@ ipv6_connect(struct TCP_Server_Info *server) ...@@ -1935,7 +1924,7 @@ ipv6_connect(struct TCP_Server_Info *server)
int rc = 0; int rc = 0;
int val; int val;
bool connected = false; bool connected = false;
__be16 orig_port = 0; bool orig_port_error = false;
struct socket *socket = server->ssocket; struct socket *socket = server->ssocket;
if (socket == NULL) { if (socket == NULL) {
...@@ -1961,13 +1950,11 @@ ipv6_connect(struct TCP_Server_Info *server) ...@@ -1961,13 +1950,11 @@ ipv6_connect(struct TCP_Server_Info *server)
sizeof(struct sockaddr_in6), 0); sizeof(struct sockaddr_in6), 0);
if (rc >= 0) if (rc >= 0)
connected = true; connected = true;
else
orig_port_error = true;
} }
if (!connected) { if (!orig_port_error && !connected) {
/* save original port so we can retry user specified port
later if fall back ports fail this time */
orig_port = server->addr.sockAddr6.sin6_port;
/* do not retry on the same port we just failed on */ /* do not retry on the same port we just failed on */
if (server->addr.sockAddr6.sin6_port != htons(CIFS_PORT)) { if (server->addr.sockAddr6.sin6_port != htons(CIFS_PORT)) {
server->addr.sockAddr6.sin6_port = htons(CIFS_PORT); server->addr.sockAddr6.sin6_port = htons(CIFS_PORT);
...@@ -1978,7 +1965,8 @@ ipv6_connect(struct TCP_Server_Info *server) ...@@ -1978,7 +1965,8 @@ ipv6_connect(struct TCP_Server_Info *server)
connected = true; connected = true;
} }
} }
if (!connected) {
if (!orig_port_error && !connected) {
server->addr.sockAddr6.sin6_port = htons(RFC1001_PORT); server->addr.sockAddr6.sin6_port = htons(RFC1001_PORT);
rc = socket->ops->connect(socket, (struct sockaddr *) rc = socket->ops->connect(socket, (struct sockaddr *)
&server->addr.sockAddr6, &server->addr.sockAddr6,
...@@ -1990,8 +1978,6 @@ ipv6_connect(struct TCP_Server_Info *server) ...@@ -1990,8 +1978,6 @@ ipv6_connect(struct TCP_Server_Info *server)
/* give up here - unless we want to retry on different /* give up here - unless we want to retry on different
protocol families some day */ protocol families some day */
if (!connected) { if (!connected) {
if (orig_port)
server->addr.sockAddr6.sin6_port = orig_port;
cFYI(1, ("Error %d connecting to server via ipv6", rc)); cFYI(1, ("Error %d connecting to server via ipv6", rc));
sock_release(socket); sock_release(socket);
server->ssocket = NULL; server->ssocket = NULL;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment