diff options
Diffstat (limited to 'drivers')
-rw-r--r-- | drivers/scsi/cxgb3i/cxgb3i.h | 2 | ||||
-rw-r--r-- | drivers/scsi/cxgb3i/cxgb3i_iscsi.c | 114 |
2 files changed, 70 insertions, 46 deletions
diff --git a/drivers/scsi/cxgb3i/cxgb3i.h b/drivers/scsi/cxgb3i/cxgb3i.h index 0942227aa7ba..d362860e7504 100644 --- a/drivers/scsi/cxgb3i/cxgb3i.h +++ b/drivers/scsi/cxgb3i/cxgb3i.h | |||
@@ -141,7 +141,7 @@ int cxgb3i_iscsi_init(void); | |||
141 | void cxgb3i_iscsi_cleanup(void); | 141 | void cxgb3i_iscsi_cleanup(void); |
142 | 142 | ||
143 | struct cxgb3i_adapter *cxgb3i_adapter_find_by_tdev(struct t3cdev *); | 143 | struct cxgb3i_adapter *cxgb3i_adapter_find_by_tdev(struct t3cdev *); |
144 | struct cxgb3i_adapter *cxgb3i_adapter_open(struct t3cdev *); | 144 | void cxgb3i_adapter_open(struct t3cdev *); |
145 | void cxgb3i_adapter_close(struct t3cdev *); | 145 | void cxgb3i_adapter_close(struct t3cdev *); |
146 | 146 | ||
147 | struct cxgb3i_hba *cxgb3i_hba_find_by_netdev(struct net_device *); | 147 | struct cxgb3i_hba *cxgb3i_hba_find_by_netdev(struct net_device *); |
diff --git a/drivers/scsi/cxgb3i/cxgb3i_iscsi.c b/drivers/scsi/cxgb3i/cxgb3i_iscsi.c index ff6bfd66733f..fff8e4327644 100644 --- a/drivers/scsi/cxgb3i/cxgb3i_iscsi.c +++ b/drivers/scsi/cxgb3i/cxgb3i_iscsi.c | |||
@@ -71,37 +71,34 @@ struct cxgb3i_adapter *cxgb3i_adapter_find_by_tdev(struct t3cdev *tdev) | |||
71 | return NULL; | 71 | return NULL; |
72 | } | 72 | } |
73 | 73 | ||
74 | /** | 74 | static inline int adapter_update(struct cxgb3i_adapter *snic) |
75 | * cxgb3i_adapter_open - init a s3 adapter structure and any h/w settings | ||
76 | * @t3dev: t3cdev adapter | ||
77 | * return the resulting cxgb3i_adapter struct | ||
78 | */ | ||
79 | struct cxgb3i_adapter *cxgb3i_adapter_open(struct t3cdev *t3dev) | ||
80 | { | 75 | { |
81 | struct cxgb3i_adapter *snic; | 76 | cxgb3i_log_info("snic 0x%p, t3dev 0x%p, updating.\n", |
82 | struct adapter *adapter = tdev2adap(t3dev); | 77 | snic, snic->tdev); |
83 | int i; | 78 | return cxgb3i_adapter_ddp_info(snic->tdev, &snic->tag_format, |
79 | &snic->tx_max_size, | ||
80 | &snic->rx_max_size); | ||
81 | } | ||
84 | 82 | ||
85 | snic = kzalloc(sizeof(*snic), GFP_KERNEL); | 83 | static int adapter_add(struct cxgb3i_adapter *snic) |
86 | if (!snic) { | 84 | { |
87 | cxgb3i_api_debug("cxgb3 %s, OOM.\n", t3dev->name); | 85 | struct t3cdev *t3dev = snic->tdev; |
88 | return NULL; | 86 | struct adapter *adapter = tdev2adap(t3dev); |
89 | } | 87 | int i, err; |
90 | spin_lock_init(&snic->lock); | ||
91 | 88 | ||
92 | snic->tdev = t3dev; | ||
93 | snic->pdev = adapter->pdev; | 89 | snic->pdev = adapter->pdev; |
94 | snic->tag_format.sw_bits = sw_tag_idx_bits + sw_tag_age_bits; | 90 | snic->tag_format.sw_bits = sw_tag_idx_bits + sw_tag_age_bits; |
95 | 91 | ||
96 | if (cxgb3i_adapter_ddp_init(t3dev, &snic->tag_format, | 92 | err = cxgb3i_adapter_ddp_info(t3dev, &snic->tag_format, |
97 | &snic->tx_max_size, | 93 | &snic->tx_max_size, |
98 | &snic->rx_max_size) < 0) | 94 | &snic->rx_max_size); |
99 | goto free_snic; | 95 | if (err < 0) |
96 | return err; | ||
100 | 97 | ||
101 | for_each_port(adapter, i) { | 98 | for_each_port(adapter, i) { |
102 | snic->hba[i] = cxgb3i_hba_host_add(snic, adapter->port[i]); | 99 | snic->hba[i] = cxgb3i_hba_host_add(snic, adapter->port[i]); |
103 | if (!snic->hba[i]) | 100 | if (!snic->hba[i]) |
104 | goto ulp_cleanup; | 101 | return -EINVAL; |
105 | } | 102 | } |
106 | snic->hba_cnt = adapter->params.nports; | 103 | snic->hba_cnt = adapter->params.nports; |
107 | 104 | ||
@@ -110,13 +107,40 @@ struct cxgb3i_adapter *cxgb3i_adapter_open(struct t3cdev *t3dev) | |||
110 | list_add_tail(&snic->list_head, &cxgb3i_snic_list); | 107 | list_add_tail(&snic->list_head, &cxgb3i_snic_list); |
111 | write_unlock(&cxgb3i_snic_rwlock); | 108 | write_unlock(&cxgb3i_snic_rwlock); |
112 | 109 | ||
113 | return snic; | 110 | cxgb3i_log_info("t3dev 0x%p open, snic 0x%p, %u scsi hosts added.\n", |
111 | t3dev, snic, snic->hba_cnt); | ||
112 | return 0; | ||
113 | } | ||
114 | 114 | ||
115 | ulp_cleanup: | 115 | /** |
116 | cxgb3i_adapter_ddp_cleanup(t3dev); | 116 | * cxgb3i_adapter_open - init a s3 adapter structure and any h/w settings |
117 | free_snic: | 117 | * @t3dev: t3cdev adapter |
118 | kfree(snic); | 118 | */ |
119 | return NULL; | 119 | void cxgb3i_adapter_open(struct t3cdev *t3dev) |
120 | { | ||
121 | struct cxgb3i_adapter *snic = cxgb3i_adapter_find_by_tdev(t3dev); | ||
122 | int err; | ||
123 | |||
124 | if (snic) | ||
125 | err = adapter_update(snic); | ||
126 | else { | ||
127 | snic = kzalloc(sizeof(*snic), GFP_KERNEL); | ||
128 | if (snic) { | ||
129 | spin_lock_init(&snic->lock); | ||
130 | snic->tdev = t3dev; | ||
131 | err = adapter_add(snic); | ||
132 | } else | ||
133 | err = -ENOMEM; | ||
134 | } | ||
135 | |||
136 | if (err < 0) { | ||
137 | cxgb3i_log_info("snic 0x%p, f 0x%x, t3dev 0x%p open, err %d.\n", | ||
138 | snic, snic ? snic->flags : 0, t3dev, err); | ||
139 | if (snic) { | ||
140 | snic->flags &= ~CXGB3I_ADAPTER_FLAG_RESET; | ||
141 | cxgb3i_adapter_close(t3dev); | ||
142 | } | ||
143 | } | ||
120 | } | 144 | } |
121 | 145 | ||
122 | /** | 146 | /** |
@@ -125,31 +149,29 @@ free_snic: | |||
125 | */ | 149 | */ |
126 | void cxgb3i_adapter_close(struct t3cdev *t3dev) | 150 | void cxgb3i_adapter_close(struct t3cdev *t3dev) |
127 | { | 151 | { |
152 | struct cxgb3i_adapter *snic = cxgb3i_adapter_find_by_tdev(t3dev); | ||
128 | int i; | 153 | int i; |
129 | struct cxgb3i_adapter *snic; | 154 | |
155 | if (!snic || snic->flags & CXGB3I_ADAPTER_FLAG_RESET) { | ||
156 | cxgb3i_log_info("t3dev 0x%p close, snic 0x%p, f 0x%x.\n", | ||
157 | t3dev, snic, snic ? snic->flags : 0); | ||
158 | return; | ||
159 | } | ||
130 | 160 | ||
131 | /* remove from the list */ | 161 | /* remove from the list */ |
132 | write_lock(&cxgb3i_snic_rwlock); | 162 | write_lock(&cxgb3i_snic_rwlock); |
133 | list_for_each_entry(snic, &cxgb3i_snic_list, list_head) { | 163 | list_del(&snic->list_head); |
134 | if (snic->tdev == t3dev) { | ||
135 | list_del(&snic->list_head); | ||
136 | break; | ||
137 | } | ||
138 | } | ||
139 | write_unlock(&cxgb3i_snic_rwlock); | 164 | write_unlock(&cxgb3i_snic_rwlock); |
140 | 165 | ||
141 | if (snic) { | 166 | for (i = 0; i < snic->hba_cnt; i++) { |
142 | for (i = 0; i < snic->hba_cnt; i++) { | 167 | if (snic->hba[i]) { |
143 | if (snic->hba[i]) { | 168 | cxgb3i_hba_host_remove(snic->hba[i]); |
144 | cxgb3i_hba_host_remove(snic->hba[i]); | 169 | snic->hba[i] = NULL; |
145 | snic->hba[i] = NULL; | ||
146 | } | ||
147 | } | 170 | } |
148 | |||
149 | /* release ddp resources */ | ||
150 | cxgb3i_adapter_ddp_cleanup(snic->tdev); | ||
151 | kfree(snic); | ||
152 | } | 171 | } |
172 | cxgb3i_log_info("t3dev 0x%p close, snic 0x%p, %u scsi hosts removed.\n", | ||
173 | t3dev, snic, snic->hba_cnt); | ||
174 | kfree(snic); | ||
153 | } | 175 | } |
154 | 176 | ||
155 | /** | 177 | /** |
@@ -189,7 +211,8 @@ struct cxgb3i_hba *cxgb3i_hba_host_add(struct cxgb3i_adapter *snic, | |||
189 | shost = iscsi_host_alloc(&cxgb3i_host_template, | 211 | shost = iscsi_host_alloc(&cxgb3i_host_template, |
190 | sizeof(struct cxgb3i_hba), 1); | 212 | sizeof(struct cxgb3i_hba), 1); |
191 | if (!shost) { | 213 | if (!shost) { |
192 | cxgb3i_log_info("iscsi_host_alloc failed.\n"); | 214 | cxgb3i_log_info("snic 0x%p, ndev 0x%p, host_alloc failed.\n", |
215 | snic, ndev); | ||
193 | return NULL; | 216 | return NULL; |
194 | } | 217 | } |
195 | 218 | ||
@@ -207,7 +230,8 @@ struct cxgb3i_hba *cxgb3i_hba_host_add(struct cxgb3i_adapter *snic, | |||
207 | pci_dev_get(snic->pdev); | 230 | pci_dev_get(snic->pdev); |
208 | err = iscsi_host_add(shost, &snic->pdev->dev); | 231 | err = iscsi_host_add(shost, &snic->pdev->dev); |
209 | if (err) { | 232 | if (err) { |
210 | cxgb3i_log_info("iscsi_host_add failed.\n"); | 233 | cxgb3i_log_info("snic 0x%p, ndev 0x%p, host_add failed.\n", |
234 | snic, ndev); | ||
211 | goto pci_dev_put; | 235 | goto pci_dev_put; |
212 | } | 236 | } |
213 | 237 | ||