| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 1 | /* | 
|  | 2 | * Copyright (c) 2009 Oracle.  All rights reserved. | 
|  | 3 | * | 
|  | 4 | * This software is available to you under a choice of one of two | 
|  | 5 | * licenses.  You may choose to be licensed under the terms of the GNU | 
|  | 6 | * General Public License (GPL) Version 2, available from the file | 
|  | 7 | * COPYING in the main directory of this source tree, or the | 
|  | 8 | * OpenIB.org BSD license below: | 
|  | 9 | * | 
|  | 10 | *     Redistribution and use in source and binary forms, with or | 
|  | 11 | *     without modification, are permitted provided that the following | 
|  | 12 | *     conditions are met: | 
|  | 13 | * | 
|  | 14 | *      - Redistributions of source code must retain the above | 
|  | 15 | *        copyright notice, this list of conditions and the following | 
|  | 16 | *        disclaimer. | 
|  | 17 | * | 
|  | 18 | *      - Redistributions in binary form must reproduce the above | 
|  | 19 | *        copyright notice, this list of conditions and the following | 
|  | 20 | *        disclaimer in the documentation and/or other materials | 
|  | 21 | *        provided with the distribution. | 
|  | 22 | * | 
|  | 23 | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, | 
|  | 24 | * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF | 
|  | 25 | * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND | 
|  | 26 | * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS | 
|  | 27 | * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN | 
|  | 28 | * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | 
|  | 29 | * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE | 
|  | 30 | * SOFTWARE. | 
|  | 31 | * | 
|  | 32 | */ | 
| Paul Gortmaker | 3a9a231 | 2011-05-27 09:12:25 -0400 | [diff] [blame] | 33 | #include <linux/module.h> | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 34 | #include <rdma/rdma_cm.h> | 
|  | 35 |  | 
|  | 36 | #include "rdma_transport.h" | 
|  | 37 |  | 
| Andy Grover | 11bc942 | 2009-04-09 14:09:37 +0000 | [diff] [blame] | 38 | static struct rdma_cm_id *rds_rdma_listen_id; | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 39 |  | 
| Zach Brown | 59f740a | 2010-08-03 13:52:47 -0700 | [diff] [blame] | 40 | static char *rds_cm_event_strings[] = { | 
|  | 41 | #define RDS_CM_EVENT_STRING(foo) \ | 
|  | 42 | [RDMA_CM_EVENT_##foo] = __stringify(RDMA_CM_EVENT_##foo) | 
|  | 43 | RDS_CM_EVENT_STRING(ADDR_RESOLVED), | 
|  | 44 | RDS_CM_EVENT_STRING(ADDR_ERROR), | 
|  | 45 | RDS_CM_EVENT_STRING(ROUTE_RESOLVED), | 
|  | 46 | RDS_CM_EVENT_STRING(ROUTE_ERROR), | 
|  | 47 | RDS_CM_EVENT_STRING(CONNECT_REQUEST), | 
|  | 48 | RDS_CM_EVENT_STRING(CONNECT_RESPONSE), | 
|  | 49 | RDS_CM_EVENT_STRING(CONNECT_ERROR), | 
|  | 50 | RDS_CM_EVENT_STRING(UNREACHABLE), | 
|  | 51 | RDS_CM_EVENT_STRING(REJECTED), | 
|  | 52 | RDS_CM_EVENT_STRING(ESTABLISHED), | 
|  | 53 | RDS_CM_EVENT_STRING(DISCONNECTED), | 
|  | 54 | RDS_CM_EVENT_STRING(DEVICE_REMOVAL), | 
|  | 55 | RDS_CM_EVENT_STRING(MULTICAST_JOIN), | 
|  | 56 | RDS_CM_EVENT_STRING(MULTICAST_ERROR), | 
|  | 57 | RDS_CM_EVENT_STRING(ADDR_CHANGE), | 
|  | 58 | RDS_CM_EVENT_STRING(TIMEWAIT_EXIT), | 
|  | 59 | #undef RDS_CM_EVENT_STRING | 
|  | 60 | }; | 
|  | 61 |  | 
|  | 62 | static char *rds_cm_event_str(enum rdma_cm_event_type type) | 
|  | 63 | { | 
|  | 64 | return rds_str_array(rds_cm_event_strings, | 
|  | 65 | ARRAY_SIZE(rds_cm_event_strings), type); | 
|  | 66 | }; | 
|  | 67 |  | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 68 | int rds_rdma_cm_event_handler(struct rdma_cm_id *cm_id, | 
|  | 69 | struct rdma_cm_event *event) | 
|  | 70 | { | 
|  | 71 | /* this can be null in the listening path */ | 
|  | 72 | struct rds_connection *conn = cm_id->context; | 
|  | 73 | struct rds_transport *trans; | 
|  | 74 | int ret = 0; | 
|  | 75 |  | 
| Zach Brown | 59f740a | 2010-08-03 13:52:47 -0700 | [diff] [blame] | 76 | rdsdebug("conn %p id %p handling event %u (%s)\n", conn, cm_id, | 
|  | 77 | event->event, rds_cm_event_str(event->event)); | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 78 |  | 
|  | 79 | if (cm_id->device->node_type == RDMA_NODE_RNIC) | 
|  | 80 | trans = &rds_iw_transport; | 
|  | 81 | else | 
|  | 82 | trans = &rds_ib_transport; | 
|  | 83 |  | 
|  | 84 | /* Prevent shutdown from tearing down the connection | 
|  | 85 | * while we're executing. */ | 
|  | 86 | if (conn) { | 
|  | 87 | mutex_lock(&conn->c_cm_lock); | 
|  | 88 |  | 
|  | 89 | /* If the connection is being shut down, bail out | 
|  | 90 | * right away. We return 0 so cm_id doesn't get | 
|  | 91 | * destroyed prematurely */ | 
|  | 92 | if (rds_conn_state(conn) == RDS_CONN_DISCONNECTING) { | 
|  | 93 | /* Reject incoming connections while we're tearing | 
|  | 94 | * down an existing one. */ | 
|  | 95 | if (event->event == RDMA_CM_EVENT_CONNECT_REQUEST) | 
|  | 96 | ret = 1; | 
|  | 97 | goto out; | 
|  | 98 | } | 
|  | 99 | } | 
|  | 100 |  | 
|  | 101 | switch (event->event) { | 
|  | 102 | case RDMA_CM_EVENT_CONNECT_REQUEST: | 
|  | 103 | ret = trans->cm_handle_connect(cm_id, event); | 
|  | 104 | break; | 
|  | 105 |  | 
|  | 106 | case RDMA_CM_EVENT_ADDR_RESOLVED: | 
|  | 107 | /* XXX do we need to clean up if this fails? */ | 
|  | 108 | ret = rdma_resolve_route(cm_id, | 
|  | 109 | RDS_RDMA_RESOLVE_TIMEOUT_MS); | 
|  | 110 | break; | 
|  | 111 |  | 
|  | 112 | case RDMA_CM_EVENT_ROUTE_RESOLVED: | 
|  | 113 | /* XXX worry about racing with listen acceptance */ | 
|  | 114 | ret = trans->cm_initiate_connect(cm_id); | 
|  | 115 | break; | 
|  | 116 |  | 
|  | 117 | case RDMA_CM_EVENT_ESTABLISHED: | 
|  | 118 | trans->cm_connect_complete(conn, event); | 
|  | 119 | break; | 
|  | 120 |  | 
|  | 121 | case RDMA_CM_EVENT_ADDR_ERROR: | 
|  | 122 | case RDMA_CM_EVENT_ROUTE_ERROR: | 
|  | 123 | case RDMA_CM_EVENT_CONNECT_ERROR: | 
|  | 124 | case RDMA_CM_EVENT_UNREACHABLE: | 
|  | 125 | case RDMA_CM_EVENT_REJECTED: | 
|  | 126 | case RDMA_CM_EVENT_DEVICE_REMOVAL: | 
|  | 127 | case RDMA_CM_EVENT_ADDR_CHANGE: | 
|  | 128 | if (conn) | 
|  | 129 | rds_conn_drop(conn); | 
|  | 130 | break; | 
|  | 131 |  | 
|  | 132 | case RDMA_CM_EVENT_DISCONNECTED: | 
| Andy Grover | 9706978 | 2010-03-11 13:50:02 +0000 | [diff] [blame] | 133 | rdsdebug("DISCONNECT event - dropping connection " | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 134 | "%pI4->%pI4\n", &conn->c_laddr, | 
|  | 135 | &conn->c_faddr); | 
|  | 136 | rds_conn_drop(conn); | 
|  | 137 | break; | 
|  | 138 |  | 
|  | 139 | default: | 
|  | 140 | /* things like device disconnect? */ | 
| Zach Brown | 59f740a | 2010-08-03 13:52:47 -0700 | [diff] [blame] | 141 | printk(KERN_ERR "RDS: unknown event %u (%s)!\n", | 
|  | 142 | event->event, rds_cm_event_str(event->event)); | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 143 | break; | 
|  | 144 | } | 
|  | 145 |  | 
|  | 146 | out: | 
|  | 147 | if (conn) | 
|  | 148 | mutex_unlock(&conn->c_cm_lock); | 
|  | 149 |  | 
| Zach Brown | 59f740a | 2010-08-03 13:52:47 -0700 | [diff] [blame] | 150 | rdsdebug("id %p event %u (%s) handling ret %d\n", cm_id, event->event, | 
|  | 151 | rds_cm_event_str(event->event), ret); | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 152 |  | 
|  | 153 | return ret; | 
|  | 154 | } | 
|  | 155 |  | 
| Zach Brown | ef87b7e | 2010-07-09 12:26:20 -0700 | [diff] [blame] | 156 | static int rds_rdma_listen_init(void) | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 157 | { | 
|  | 158 | struct sockaddr_in sin; | 
|  | 159 | struct rdma_cm_id *cm_id; | 
|  | 160 | int ret; | 
|  | 161 |  | 
| Sean Hefty | b26f9b9 | 2010-04-01 17:08:41 +0000 | [diff] [blame] | 162 | cm_id = rdma_create_id(rds_rdma_cm_event_handler, NULL, RDMA_PS_TCP, | 
|  | 163 | IB_QPT_RC); | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 164 | if (IS_ERR(cm_id)) { | 
|  | 165 | ret = PTR_ERR(cm_id); | 
| Andy Grover | 92c330b | 2009-07-17 13:13:26 +0000 | [diff] [blame] | 166 | printk(KERN_ERR "RDS/RDMA: failed to setup listener, " | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 167 | "rdma_create_id() returned %d\n", ret); | 
| Dan Carpenter | 24acc68 | 2010-04-21 23:55:27 +0000 | [diff] [blame] | 168 | return ret; | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 169 | } | 
|  | 170 |  | 
| Julia Lawall | 3d7ddd5 | 2009-08-05 20:30:13 -0700 | [diff] [blame] | 171 | sin.sin_family = AF_INET, | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 172 | sin.sin_addr.s_addr = (__force u32)htonl(INADDR_ANY); | 
|  | 173 | sin.sin_port = (__force u16)htons(RDS_PORT); | 
|  | 174 |  | 
|  | 175 | /* | 
|  | 176 | * XXX I bet this binds the cm_id to a device.  If we want to support | 
|  | 177 | * fail-over we'll have to take this into consideration. | 
|  | 178 | */ | 
|  | 179 | ret = rdma_bind_addr(cm_id, (struct sockaddr *)&sin); | 
|  | 180 | if (ret) { | 
| Andy Grover | 92c330b | 2009-07-17 13:13:26 +0000 | [diff] [blame] | 181 | printk(KERN_ERR "RDS/RDMA: failed to setup listener, " | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 182 | "rdma_bind_addr() returned %d\n", ret); | 
|  | 183 | goto out; | 
|  | 184 | } | 
|  | 185 |  | 
|  | 186 | ret = rdma_listen(cm_id, 128); | 
|  | 187 | if (ret) { | 
| Andy Grover | 92c330b | 2009-07-17 13:13:26 +0000 | [diff] [blame] | 188 | printk(KERN_ERR "RDS/RDMA: failed to setup listener, " | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 189 | "rdma_listen() returned %d\n", ret); | 
|  | 190 | goto out; | 
|  | 191 | } | 
|  | 192 |  | 
|  | 193 | rdsdebug("cm %p listening on port %u\n", cm_id, RDS_PORT); | 
|  | 194 |  | 
| Andy Grover | 11bc942 | 2009-04-09 14:09:37 +0000 | [diff] [blame] | 195 | rds_rdma_listen_id = cm_id; | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 196 | cm_id = NULL; | 
|  | 197 | out: | 
|  | 198 | if (cm_id) | 
|  | 199 | rdma_destroy_id(cm_id); | 
|  | 200 | return ret; | 
|  | 201 | } | 
|  | 202 |  | 
|  | 203 | static void rds_rdma_listen_stop(void) | 
|  | 204 | { | 
| Andy Grover | 11bc942 | 2009-04-09 14:09:37 +0000 | [diff] [blame] | 205 | if (rds_rdma_listen_id) { | 
|  | 206 | rdsdebug("cm %p\n", rds_rdma_listen_id); | 
|  | 207 | rdma_destroy_id(rds_rdma_listen_id); | 
|  | 208 | rds_rdma_listen_id = NULL; | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 209 | } | 
|  | 210 | } | 
|  | 211 |  | 
| stephen hemminger | ff51bf8 | 2010-10-19 08:08:33 +0000 | [diff] [blame] | 212 | static int rds_rdma_init(void) | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 213 | { | 
|  | 214 | int ret; | 
|  | 215 |  | 
|  | 216 | ret = rds_rdma_listen_init(); | 
|  | 217 | if (ret) | 
|  | 218 | goto out; | 
|  | 219 |  | 
|  | 220 | ret = rds_iw_init(); | 
|  | 221 | if (ret) | 
|  | 222 | goto err_iw_init; | 
|  | 223 |  | 
|  | 224 | ret = rds_ib_init(); | 
|  | 225 | if (ret) | 
|  | 226 | goto err_ib_init; | 
|  | 227 |  | 
|  | 228 | goto out; | 
|  | 229 |  | 
|  | 230 | err_ib_init: | 
|  | 231 | rds_iw_exit(); | 
|  | 232 | err_iw_init: | 
|  | 233 | rds_rdma_listen_stop(); | 
|  | 234 | out: | 
|  | 235 | return ret; | 
|  | 236 | } | 
| Andy Grover | 40d8660 | 2009-08-21 12:28:33 +0000 | [diff] [blame] | 237 | module_init(rds_rdma_init); | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 238 |  | 
| stephen hemminger | ff51bf8 | 2010-10-19 08:08:33 +0000 | [diff] [blame] | 239 | static void rds_rdma_exit(void) | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 240 | { | 
|  | 241 | /* stop listening first to ensure no new connections are attempted */ | 
|  | 242 | rds_rdma_listen_stop(); | 
|  | 243 | rds_ib_exit(); | 
|  | 244 | rds_iw_exit(); | 
|  | 245 | } | 
| Andy Grover | 40d8660 | 2009-08-21 12:28:33 +0000 | [diff] [blame] | 246 | module_exit(rds_rdma_exit); | 
|  | 247 |  | 
|  | 248 | MODULE_AUTHOR("Oracle Corporation <rds-devel@oss.oracle.com>"); | 
|  | 249 | MODULE_DESCRIPTION("RDS: IB/iWARP transport"); | 
|  | 250 | MODULE_LICENSE("Dual BSD/GPL"); | 
| Andy Grover | 55b7ed0 | 2009-02-24 15:30:37 +0000 | [diff] [blame] | 251 |  |