aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/pcmcia
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/pcmcia')
-rw-r--r--drivers/pcmcia/cs_internal.h5
-rw-r--r--drivers/pcmcia/pcmcia_ioctl.c57
-rw-r--r--drivers/pcmcia/pcmcia_resource.c39
3 files changed, 58 insertions, 43 deletions
diff --git a/drivers/pcmcia/cs_internal.h b/drivers/pcmcia/cs_internal.h
index f889a44280f5..88c96aee0f45 100644
--- a/drivers/pcmcia/cs_internal.h
+++ b/drivers/pcmcia/cs_internal.h
@@ -126,10 +126,9 @@ extern struct class_interface pccard_sysfs_interface;
126extern struct rw_semaphore pcmcia_socket_list_rwsem; 126extern struct rw_semaphore pcmcia_socket_list_rwsem;
127extern struct list_head pcmcia_socket_list; 127extern struct list_head pcmcia_socket_list;
128int pcmcia_get_window(struct pcmcia_socket *s, window_handle_t *handle, int idx, win_req_t *req); 128int pcmcia_get_window(struct pcmcia_socket *s, window_handle_t *handle, int idx, win_req_t *req);
129int pccard_get_configuration_info(struct pcmcia_socket *s, unsigned int function, config_info_t *config); 129int pccard_get_configuration_info(struct pcmcia_socket *s, struct pcmcia_device *p_dev, config_info_t *config);
130int pccard_reset_card(struct pcmcia_socket *skt); 130int pccard_reset_card(struct pcmcia_socket *skt);
131int pccard_get_status(struct pcmcia_socket *s, unsigned int function, cs_status_t *status); 131int pccard_get_status(struct pcmcia_socket *s, struct pcmcia_device *p_dev, cs_status_t *status);
132int pccard_access_configuration_register(struct pcmcia_socket *s, unsigned int function, conf_reg_t *reg);
133 132
134 133
135struct pcmcia_callback{ 134struct pcmcia_callback{
diff --git a/drivers/pcmcia/pcmcia_ioctl.c b/drivers/pcmcia/pcmcia_ioctl.c
index 80969f7e7a0b..56b625d171ae 100644
--- a/drivers/pcmcia/pcmcia_ioctl.c
+++ b/drivers/pcmcia/pcmcia_ioctl.c
@@ -70,10 +70,26 @@ extern int ds_pc_debug;
70#define ds_dbg(lvl, fmt, arg...) do { } while (0) 70#define ds_dbg(lvl, fmt, arg...) do { } while (0)
71#endif 71#endif
72 72
73static struct pcmcia_device *get_pcmcia_device(struct pcmcia_socket *s,
74 unsigned int function)
75{
76 struct pcmcia_device *p_dev = NULL;
77 unsigned long flags;
78
79 spin_lock_irqsave(&pcmcia_dev_list_lock, flags);
80 list_for_each_entry(p_dev, &s->devices_list, socket_device_list) {
81 if (p_dev->func == function) {
82 spin_unlock_irqrestore(&pcmcia_dev_list_lock, flags);
83 return pcmcia_get_dev(p_dev);
84 }
85 }
86 spin_unlock_irqrestore(&pcmcia_dev_list_lock, flags);
87 return NULL;
88}
73 89
74/* backwards-compatible accessing of driver --- by name! */ 90/* backwards-compatible accessing of driver --- by name! */
75 91
76static struct pcmcia_driver * get_pcmcia_driver (dev_info_t *dev_info) 92static struct pcmcia_driver *get_pcmcia_driver(dev_info_t *dev_info)
77{ 93{
78 struct device_driver *drv; 94 struct device_driver *drv;
79 struct pcmcia_driver *p_drv; 95 struct pcmcia_driver *p_drv;
@@ -583,9 +599,11 @@ static int ds_ioctl(struct inode * inode, struct file * file,
583 if (buf->config.Function && 599 if (buf->config.Function &&
584 (buf->config.Function >= s->functions)) 600 (buf->config.Function >= s->functions))
585 ret = CS_BAD_ARGS; 601 ret = CS_BAD_ARGS;
586 else 602 else {
587 ret = pccard_get_configuration_info(s, 603 struct pcmcia_device *p_dev = get_pcmcia_device(s, buf->config.Function);
588 buf->config.Function, &buf->config); 604 ret = pccard_get_configuration_info(s, p_dev, &buf->config);
605 pcmcia_put_dev(p_dev);
606 }
589 break; 607 break;
590 case DS_GET_FIRST_TUPLE: 608 case DS_GET_FIRST_TUPLE:
591 down(&s->skt_sem); 609 down(&s->skt_sem);
@@ -609,12 +627,15 @@ static int ds_ioctl(struct inode * inode, struct file * file,
609 ret = pccard_reset_card(s); 627 ret = pccard_reset_card(s);
610 break; 628 break;
611 case DS_GET_STATUS: 629 case DS_GET_STATUS:
612 if (buf->status.Function && 630 if (buf->status.Function &&
613 (buf->status.Function >= s->functions)) 631 (buf->status.Function >= s->functions))
614 ret = CS_BAD_ARGS; 632 ret = CS_BAD_ARGS;
615 else 633 else {
616 ret = pccard_get_status(s, buf->status.Function, &buf->status); 634 struct pcmcia_device *p_dev = get_pcmcia_device(s, buf->status.Function);
617 break; 635 ret = pccard_get_status(s, p_dev, &buf->status);
636 pcmcia_put_dev(p_dev);
637 }
638 break;
618 case DS_VALIDATE_CIS: 639 case DS_VALIDATE_CIS:
619 down(&s->skt_sem); 640 down(&s->skt_sem);
620 pcmcia_validate_mem(s); 641 pcmcia_validate_mem(s);
@@ -638,12 +659,16 @@ static int ds_ioctl(struct inode * inode, struct file * file,
638 err = -EPERM; 659 err = -EPERM;
639 goto free_out; 660 goto free_out;
640 } 661 }
641 if (buf->conf_reg.Function && 662
642 (buf->conf_reg.Function >= s->functions)) 663 ret = CS_BAD_ARGS;
643 ret = CS_BAD_ARGS; 664
644 else 665 if (!(buf->conf_reg.Function &&
645 ret = pccard_access_configuration_register(s, 666 (buf->conf_reg.Function >= s->functions))) {
646 buf->conf_reg.Function, &buf->conf_reg); 667 struct pcmcia_device *p_dev = get_pcmcia_device(s, buf->conf_reg.Function);
668 if (p_dev)
669 ret = pcmcia_access_configuration_register(p_dev, &buf->conf_reg);
670 pcmcia_put_dev(p_dev);
671 }
647 break; 672 break;
648 case DS_GET_FIRST_REGION: 673 case DS_GET_FIRST_REGION:
649 case DS_GET_NEXT_REGION: 674 case DS_GET_NEXT_REGION:
diff --git a/drivers/pcmcia/pcmcia_resource.c b/drivers/pcmcia/pcmcia_resource.c
index 11a94d9a15fb..aabde8be6470 100644
--- a/drivers/pcmcia/pcmcia_resource.c
+++ b/drivers/pcmcia/pcmcia_resource.c
@@ -165,21 +165,19 @@ static void release_io_space(struct pcmcia_socket *s, ioaddr_t base,
165 * this and the tuple reading services. 165 * this and the tuple reading services.
166 */ 166 */
167 167
168int pccard_access_configuration_register(struct pcmcia_socket *s, 168int pcmcia_access_configuration_register(struct pcmcia_device *p_dev,
169 unsigned int function,
170 conf_reg_t *reg) 169 conf_reg_t *reg)
171{ 170{
171 struct pcmcia_socket *s;
172 config_t *c; 172 config_t *c;
173 int addr; 173 int addr;
174 u_char val; 174 u_char val;
175 175
176 if (!s || !s->config) 176 if (!p_dev || !p_dev->function_config)
177 return CS_NO_CARD; 177 return CS_NO_CARD;
178 178
179 c = &s->config[function]; 179 s = p_dev->socket;
180 180 c = p_dev->function_config;
181 if (c == NULL)
182 return CS_NO_CARD;
183 181
184 if (!(c->state & CONFIG_LOCKED)) 182 if (!(c->state & CONFIG_LOCKED))
185 return CS_CONFIGURATION_LOCKED; 183 return CS_CONFIGURATION_LOCKED;
@@ -200,20 +198,12 @@ int pccard_access_configuration_register(struct pcmcia_socket *s,
200 break; 198 break;
201 } 199 }
202 return CS_SUCCESS; 200 return CS_SUCCESS;
203} /* pccard_access_configuration_register */ 201} /* pcmcia_access_configuration_register */
204
205int pcmcia_access_configuration_register(struct pcmcia_device *p_dev,
206 conf_reg_t *reg)
207{
208 return pccard_access_configuration_register(p_dev->socket,
209 p_dev->func, reg);
210}
211EXPORT_SYMBOL(pcmcia_access_configuration_register); 202EXPORT_SYMBOL(pcmcia_access_configuration_register);
212 203
213 204
214
215int pccard_get_configuration_info(struct pcmcia_socket *s, 205int pccard_get_configuration_info(struct pcmcia_socket *s,
216 unsigned int function, 206 struct pcmcia_device *p_dev,
217 config_info_t *config) 207 config_info_t *config)
218{ 208{
219 config_t *c; 209 config_t *c;
@@ -221,7 +211,7 @@ int pccard_get_configuration_info(struct pcmcia_socket *s,
221 if (!(s->state & SOCKET_PRESENT)) 211 if (!(s->state & SOCKET_PRESENT))
222 return CS_NO_CARD; 212 return CS_NO_CARD;
223 213
224 config->Function = function; 214 config->Function = p_dev->func;
225 215
226#ifdef CONFIG_CARDBUS 216#ifdef CONFIG_CARDBUS
227 if (s->state & SOCKET_CARDBUS) { 217 if (s->state & SOCKET_CARDBUS) {
@@ -242,7 +232,7 @@ int pccard_get_configuration_info(struct pcmcia_socket *s,
242 } 232 }
243#endif 233#endif
244 234
245 c = (s->config != NULL) ? &s->config[function] : NULL; 235 c = (p_dev) ? p_dev->function_config : NULL;
246 236
247 if ((c == NULL) || !(c->state & CONFIG_LOCKED)) { 237 if ((c == NULL) || !(c->state & CONFIG_LOCKED)) {
248 config->Attributes = 0; 238 config->Attributes = 0;
@@ -271,7 +261,7 @@ int pccard_get_configuration_info(struct pcmcia_socket *s,
271int pcmcia_get_configuration_info(struct pcmcia_device *p_dev, 261int pcmcia_get_configuration_info(struct pcmcia_device *p_dev,
272 config_info_t *config) 262 config_info_t *config)
273{ 263{
274 return pccard_get_configuration_info(p_dev->socket, p_dev->func, 264 return pccard_get_configuration_info(p_dev->socket, p_dev,
275 config); 265 config);
276} 266}
277EXPORT_SYMBOL(pcmcia_get_configuration_info); 267EXPORT_SYMBOL(pcmcia_get_configuration_info);
@@ -317,7 +307,7 @@ EXPORT_SYMBOL(pcmcia_get_window);
317 * SocketState yet: I haven't seen any point for it. 307 * SocketState yet: I haven't seen any point for it.
318 */ 308 */
319 309
320int pccard_get_status(struct pcmcia_socket *s, unsigned int function, 310int pccard_get_status(struct pcmcia_socket *s, struct pcmcia_device *p_dev,
321 cs_status_t *status) 311 cs_status_t *status)
322{ 312{
323 config_t *c; 313 config_t *c;
@@ -334,7 +324,8 @@ int pccard_get_status(struct pcmcia_socket *s, unsigned int function,
334 if (!(s->state & SOCKET_PRESENT)) 324 if (!(s->state & SOCKET_PRESENT))
335 return CS_NO_CARD; 325 return CS_NO_CARD;
336 326
337 c = (s->config != NULL) ? &s->config[function] : NULL; 327 c = (p_dev) ? p_dev->function_config : NULL;
328
338 if ((c != NULL) && (c->state & CONFIG_LOCKED) && 329 if ((c != NULL) && (c->state & CONFIG_LOCKED) &&
339 (c->IntType & (INT_MEMORY_AND_IO | INT_ZOOMED_VIDEO))) { 330 (c->IntType & (INT_MEMORY_AND_IO | INT_ZOOMED_VIDEO))) {
340 u_char reg; 331 u_char reg;
@@ -370,9 +361,9 @@ int pccard_get_status(struct pcmcia_socket *s, unsigned int function,
370 return CS_SUCCESS; 361 return CS_SUCCESS;
371} /* pccard_get_status */ 362} /* pccard_get_status */
372 363
373int pcmcia_get_status(client_handle_t handle, cs_status_t *status) 364int pcmcia_get_status(struct pcmcia_device *p_dev, cs_status_t *status)
374{ 365{
375 return pccard_get_status(handle->socket, handle->func, status); 366 return pccard_get_status(p_dev->socket, p_dev, status);
376} 367}
377EXPORT_SYMBOL(pcmcia_get_status); 368EXPORT_SYMBOL(pcmcia_get_status);
378 369