aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorDominik Brodowski <linux@dominikbrodowski.net>2010-01-17 13:31:45 -0500
committerDominik Brodowski <linux@dominikbrodowski.net>2010-02-17 11:48:24 -0500
commitcfe5d809518eda3d5e2da87c5ccbe8647143573a (patch)
treedd61ec285412fc0de8090ef5743b824447a9eeda
parent3f565232c561fbd9d5e03354aac29b90cb2bc78a (diff)
pcmcia: use ops_mutex for rsrc_{mgr,nonstatic} locking
Tested-by: Wolfram Sang <w.sang@pengutronix.de> Signed-off-by: Dominik Brodowski <linux@dominikbrodowski.net>
-rw-r--r--Documentation/pcmcia/locking.txt5
-rw-r--r--drivers/pcmcia/cs.c7
-rw-r--r--drivers/pcmcia/ds.c6
-rw-r--r--drivers/pcmcia/pcmcia_ioctl.c4
-rw-r--r--drivers/pcmcia/rsrc_mgr.c2
-rw-r--r--drivers/pcmcia/rsrc_nonstatic.c35
6 files changed, 23 insertions, 36 deletions
diff --git a/Documentation/pcmcia/locking.txt b/Documentation/pcmcia/locking.txt
index d6251056128..68f622bc406 100644
--- a/Documentation/pcmcia/locking.txt
+++ b/Documentation/pcmcia/locking.txt
@@ -36,7 +36,8 @@ be called with "ops_mutex" held:
36 socket_reset() 36 socket_reset()
37 socket_setup() 37 socket_setup()
38 38
39 struct pccard_operations *ops 39 struct pccard_operations *ops
40 struct pccard_resource_ops *resource_ops;
40 41
41Note that send_event() and struct pcmcia_callback *callback must not be 42Note that send_event() and struct pcmcia_callback *callback must not be
42called with "ops_mutex" held. 43called with "ops_mutex" held.
@@ -54,7 +55,7 @@ protected by pcmcia_socket_list_rwsem;
54 55
552. Per-Socket Data: 562. Per-Socket Data:
56------------------- 57-------------------
57The resource_ops are on their own to provide proper locking. 58The resource_ops and their data are protected by ops_mutex.
58 59
59The "main" struct pcmcia_socket is protected as follows (read-only fields 60The "main" struct pcmcia_socket is protected as follows (read-only fields
60or single-use fields not mentioned): 61or single-use fields not mentioned):
diff --git a/drivers/pcmcia/cs.c b/drivers/pcmcia/cs.c
index 13277eebe46..7ba45b0cca6 100644
--- a/drivers/pcmcia/cs.c
+++ b/drivers/pcmcia/cs.c
@@ -224,7 +224,9 @@ int pcmcia_register_socket(struct pcmcia_socket *socket)
224 spin_lock_init(&socket->thread_lock); 224 spin_lock_init(&socket->thread_lock);
225 225
226 if (socket->resource_ops->init) { 226 if (socket->resource_ops->init) {
227 mutex_lock(&socket->ops_mutex);
227 ret = socket->resource_ops->init(socket); 228 ret = socket->resource_ops->init(socket);
229 mutex_unlock(&socket->ops_mutex);
228 if (ret) 230 if (ret)
229 goto err; 231 goto err;
230 } 232 }
@@ -282,8 +284,11 @@ void pcmcia_unregister_socket(struct pcmcia_socket *socket)
282 up_write(&pcmcia_socket_list_rwsem); 284 up_write(&pcmcia_socket_list_rwsem);
283 285
284 /* wait for sysfs to drop all references */ 286 /* wait for sysfs to drop all references */
285 if (socket->resource_ops->exit) 287 if (socket->resource_ops->exit) {
288 mutex_lock(&socket->ops_mutex);
286 socket->resource_ops->exit(socket); 289 socket->resource_ops->exit(socket);
290 mutex_unlock(&socket->ops_mutex);
291 }
287 wait_for_completion(&socket->socket_released); 292 wait_for_completion(&socket->socket_released);
288} /* pcmcia_unregister_socket */ 293} /* pcmcia_unregister_socket */
289EXPORT_SYMBOL(pcmcia_unregister_socket); 294EXPORT_SYMBOL(pcmcia_unregister_socket);
diff --git a/drivers/pcmcia/ds.c b/drivers/pcmcia/ds.c
index 253d9aca5f7..76a21638291 100644
--- a/drivers/pcmcia/ds.c
+++ b/drivers/pcmcia/ds.c
@@ -607,19 +607,23 @@ static int pcmcia_card_add(struct pcmcia_socket *s)
607{ 607{
608 cistpl_longlink_mfc_t mfc; 608 cistpl_longlink_mfc_t mfc;
609 unsigned int no_funcs, i, no_chains; 609 unsigned int no_funcs, i, no_chains;
610 int ret = 0; 610 int ret = -EAGAIN;
611 611
612 mutex_lock(&s->ops_mutex);
612 if (!(s->resource_setup_done)) { 613 if (!(s->resource_setup_done)) {
613 dev_dbg(&s->dev, 614 dev_dbg(&s->dev,
614 "no resources available, delaying card_add\n"); 615 "no resources available, delaying card_add\n");
616 mutex_unlock(&s->ops_mutex);
615 return -EAGAIN; /* try again, but later... */ 617 return -EAGAIN; /* try again, but later... */
616 } 618 }
617 619
618 if (pcmcia_validate_mem(s)) { 620 if (pcmcia_validate_mem(s)) {
619 dev_dbg(&s->dev, "validating mem resources failed, " 621 dev_dbg(&s->dev, "validating mem resources failed, "
620 "delaying card_add\n"); 622 "delaying card_add\n");
623 mutex_unlock(&s->ops_mutex);
621 return -EAGAIN; /* try again, but later... */ 624 return -EAGAIN; /* try again, but later... */
622 } 625 }
626 mutex_unlock(&s->ops_mutex);
623 627
624 ret = pccard_validate_cis(s, &no_chains); 628 ret = pccard_validate_cis(s, &no_chains);
625 if (ret || !no_chains) { 629 if (ret || !no_chains) {
diff --git a/drivers/pcmcia/pcmcia_ioctl.c b/drivers/pcmcia/pcmcia_ioctl.c
index db2e3db8008..96fd236f52a 100644
--- a/drivers/pcmcia/pcmcia_ioctl.c
+++ b/drivers/pcmcia/pcmcia_ioctl.c
@@ -187,7 +187,6 @@ static int pcmcia_adjust_resource_info(adjust_t *adj)
187 continue; 187 continue;
188 } else if (!(s->resource_setup_old)) 188 } else if (!(s->resource_setup_old))
189 s->resource_setup_old = 1; 189 s->resource_setup_old = 1;
190 mutex_unlock(&s->ops_mutex);
191 190
192 switch (adj->Resource) { 191 switch (adj->Resource) {
193 case RES_MEMORY_RANGE: 192 case RES_MEMORY_RANGE:
@@ -206,10 +205,9 @@ static int pcmcia_adjust_resource_info(adjust_t *adj)
206 * last call to adjust_resource_info, we 205 * last call to adjust_resource_info, we
207 * always need to assume this is the latest 206 * always need to assume this is the latest
208 * one... */ 207 * one... */
209 mutex_lock(&s->ops_mutex);
210 s->resource_setup_done = 1; 208 s->resource_setup_done = 1;
211 mutex_unlock(&s->ops_mutex);
212 } 209 }
210 mutex_unlock(&s->ops_mutex);
213 } 211 }
214 } 212 }
215 up_read(&pcmcia_socket_list_rwsem); 213 up_read(&pcmcia_socket_list_rwsem);
diff --git a/drivers/pcmcia/rsrc_mgr.c b/drivers/pcmcia/rsrc_mgr.c
index b81586622d0..aca2cfd02ca 100644
--- a/drivers/pcmcia/rsrc_mgr.c
+++ b/drivers/pcmcia/rsrc_mgr.c
@@ -26,9 +26,7 @@ static int static_init(struct pcmcia_socket *s)
26 /* the good thing about SS_CAP_STATIC_MAP sockets is 26 /* the good thing about SS_CAP_STATIC_MAP sockets is
27 * that they don't need a resource database */ 27 * that they don't need a resource database */
28 28
29 mutex_lock(&s->ops_mutex);
30 s->resource_setup_done = 1; 29 s->resource_setup_done = 1;
31 mutex_unlock(&s->ops_mutex);
32 30
33 return 0; 31 return 0;
34} 32}
diff --git a/drivers/pcmcia/rsrc_nonstatic.c b/drivers/pcmcia/rsrc_nonstatic.c
index 1de46cf2772..c13424f7b47 100644
--- a/drivers/pcmcia/rsrc_nonstatic.c
+++ b/drivers/pcmcia/rsrc_nonstatic.c
@@ -59,7 +59,6 @@ struct socket_data {
59 unsigned int rsrc_mem_probe; 59 unsigned int rsrc_mem_probe;
60}; 60};
61 61
62static DEFINE_MUTEX(rsrc_mutex);
63#define MEM_PROBE_LOW (1 << 0) 62#define MEM_PROBE_LOW (1 << 0)
64#define MEM_PROBE_HIGH (1 << 1) 63#define MEM_PROBE_HIGH (1 << 1)
65 64
@@ -274,7 +273,6 @@ static int readable(struct pcmcia_socket *s, struct resource *res,
274{ 273{
275 int ret = -EINVAL; 274 int ret = -EINVAL;
276 275
277 mutex_lock(&s->ops_mutex);
278 s->cis_mem.res = res; 276 s->cis_mem.res = res;
279 s->cis_virt = ioremap(res->start, s->map_size); 277 s->cis_virt = ioremap(res->start, s->map_size);
280 if (s->cis_virt) { 278 if (s->cis_virt) {
@@ -288,7 +286,6 @@ static int readable(struct pcmcia_socket *s, struct resource *res,
288 s->cis_virt = NULL; 286 s->cis_virt = NULL;
289 } 287 }
290 s->cis_mem.res = NULL; 288 s->cis_mem.res = NULL;
291 mutex_unlock(&s->ops_mutex);
292 if ((ret) || (*count == 0)) 289 if ((ret) || (*count == 0))
293 return -EINVAL; 290 return -EINVAL;
294 return 0; 291 return 0;
@@ -304,8 +301,6 @@ static int checksum(struct pcmcia_socket *s, struct resource *res,
304 int i, a = 0, b = -1, d; 301 int i, a = 0, b = -1, d;
305 void __iomem *virt; 302 void __iomem *virt;
306 303
307 mutex_lock(&s->ops_mutex);
308
309 virt = ioremap(res->start, s->map_size); 304 virt = ioremap(res->start, s->map_size);
310 if (virt) { 305 if (virt) {
311 map.map = 0; 306 map.map = 0;
@@ -328,8 +323,6 @@ static int checksum(struct pcmcia_socket *s, struct resource *res,
328 iounmap(virt); 323 iounmap(virt);
329 } 324 }
330 325
331 mutex_unlock(&s->ops_mutex);
332
333 if (b == -1) 326 if (b == -1)
334 return -EINVAL; 327 return -EINVAL;
335 328
@@ -570,8 +563,6 @@ static int pcmcia_nonstatic_validate_mem(struct pcmcia_socket *s)
570 if (!probe_mem) 563 if (!probe_mem)
571 return 0; 564 return 0;
572 565
573 mutex_lock(&rsrc_mutex);
574
575 if (s->features & SS_CAP_PAGE_REGS) 566 if (s->features & SS_CAP_PAGE_REGS)
576 probe_mask = MEM_PROBE_HIGH; 567 probe_mask = MEM_PROBE_HIGH;
577 568
@@ -583,8 +574,6 @@ static int pcmcia_nonstatic_validate_mem(struct pcmcia_socket *s)
583 } 574 }
584 } 575 }
585 576
586 mutex_unlock(&rsrc_mutex);
587
588 return ret; 577 return ret;
589} 578}
590 579
@@ -661,7 +650,6 @@ static int nonstatic_adjust_io_region(struct resource *res, unsigned long r_star
661 struct socket_data *s_data = s->resource_data; 650 struct socket_data *s_data = s->resource_data;
662 int ret = -ENOMEM; 651 int ret = -ENOMEM;
663 652
664 mutex_lock(&rsrc_mutex);
665 for (m = s_data->io_db.next; m != &s_data->io_db; m = m->next) { 653 for (m = s_data->io_db.next; m != &s_data->io_db; m = m->next) {
666 unsigned long start = m->base; 654 unsigned long start = m->base;
667 unsigned long end = m->base + m->num - 1; 655 unsigned long end = m->base + m->num - 1;
@@ -672,7 +660,6 @@ static int nonstatic_adjust_io_region(struct resource *res, unsigned long r_star
672 ret = adjust_resource(res, r_start, r_end - r_start + 1); 660 ret = adjust_resource(res, r_start, r_end - r_start + 1);
673 break; 661 break;
674 } 662 }
675 mutex_unlock(&rsrc_mutex);
676 663
677 return ret; 664 return ret;
678} 665}
@@ -706,7 +693,6 @@ static struct resource *nonstatic_find_io_region(unsigned long base, int num,
706 data.offset = base & data.mask; 693 data.offset = base & data.mask;
707 data.map = &s_data->io_db; 694 data.map = &s_data->io_db;
708 695
709 mutex_lock(&rsrc_mutex);
710#ifdef CONFIG_PCI 696#ifdef CONFIG_PCI
711 if (s->cb_dev) { 697 if (s->cb_dev) {
712 ret = pci_bus_alloc_resource(s->cb_dev->bus, res, num, 1, 698 ret = pci_bus_alloc_resource(s->cb_dev->bus, res, num, 1,
@@ -715,7 +701,6 @@ static struct resource *nonstatic_find_io_region(unsigned long base, int num,
715#endif 701#endif
716 ret = allocate_resource(&ioport_resource, res, num, min, ~0UL, 702 ret = allocate_resource(&ioport_resource, res, num, min, ~0UL,
717 1, pcmcia_align, &data); 703 1, pcmcia_align, &data);
718 mutex_unlock(&rsrc_mutex);
719 704
720 if (ret != 0) { 705 if (ret != 0) {
721 kfree(res); 706 kfree(res);
@@ -748,7 +733,6 @@ static struct resource *nonstatic_find_mem_region(u_long base, u_long num,
748 min = 0x100000UL + base; 733 min = 0x100000UL + base;
749 } 734 }
750 735
751 mutex_lock(&rsrc_mutex);
752#ifdef CONFIG_PCI 736#ifdef CONFIG_PCI
753 if (s->cb_dev) { 737 if (s->cb_dev) {
754 ret = pci_bus_alloc_resource(s->cb_dev->bus, res, num, 738 ret = pci_bus_alloc_resource(s->cb_dev->bus, res, num,
@@ -758,7 +742,6 @@ static struct resource *nonstatic_find_mem_region(u_long base, u_long num,
758#endif 742#endif
759 ret = allocate_resource(&iomem_resource, res, num, min, 743 ret = allocate_resource(&iomem_resource, res, num, min,
760 max, 1, pcmcia_align, &data); 744 max, 1, pcmcia_align, &data);
761 mutex_unlock(&rsrc_mutex);
762 if (ret == 0 || low) 745 if (ret == 0 || low)
763 break; 746 break;
764 low = 1; 747 low = 1;
@@ -781,7 +764,6 @@ static int adjust_memory(struct pcmcia_socket *s, unsigned int action, unsigned
781 if (end < start) 764 if (end < start)
782 return -EINVAL; 765 return -EINVAL;
783 766
784 mutex_lock(&rsrc_mutex);
785 switch (action) { 767 switch (action) {
786 case ADD_MANAGED_RESOURCE: 768 case ADD_MANAGED_RESOURCE:
787 ret = add_interval(&data->mem_db, start, size); 769 ret = add_interval(&data->mem_db, start, size);
@@ -794,7 +776,6 @@ static int adjust_memory(struct pcmcia_socket *s, unsigned int action, unsigned
794 default: 776 default:
795 ret = -EINVAL; 777 ret = -EINVAL;
796 } 778 }
797 mutex_unlock(&rsrc_mutex);
798 779
799 return ret; 780 return ret;
800} 781}
@@ -812,7 +793,6 @@ static int adjust_io(struct pcmcia_socket *s, unsigned int action, unsigned long
812 if (end > IO_SPACE_LIMIT) 793 if (end > IO_SPACE_LIMIT)
813 return -EINVAL; 794 return -EINVAL;
814 795
815 mutex_lock(&rsrc_mutex);
816 switch (action) { 796 switch (action) {
817 case ADD_MANAGED_RESOURCE: 797 case ADD_MANAGED_RESOURCE:
818 if (add_interval(&data->io_db, start, size) != 0) { 798 if (add_interval(&data->io_db, start, size) != 0) {
@@ -831,7 +811,6 @@ static int adjust_io(struct pcmcia_socket *s, unsigned int action, unsigned long
831 ret = -EINVAL; 811 ret = -EINVAL;
832 break; 812 break;
833 } 813 }
834 mutex_unlock(&rsrc_mutex);
835 814
836 return ret; 815 return ret;
837} 816}
@@ -929,7 +908,6 @@ static void nonstatic_release_resource_db(struct pcmcia_socket *s)
929 struct socket_data *data = s->resource_data; 908 struct socket_data *data = s->resource_data;
930 struct resource_map *p, *q; 909 struct resource_map *p, *q;
931 910
932 mutex_lock(&rsrc_mutex);
933 for (p = data->mem_db.next; p != &data->mem_db; p = q) { 911 for (p = data->mem_db.next; p != &data->mem_db; p = q) {
934 q = p->next; 912 q = p->next;
935 kfree(p); 913 kfree(p);
@@ -938,7 +916,6 @@ static void nonstatic_release_resource_db(struct pcmcia_socket *s)
938 q = p->next; 916 q = p->next;
939 kfree(p); 917 kfree(p);
940 } 918 }
941 mutex_unlock(&rsrc_mutex);
942} 919}
943 920
944 921
@@ -965,7 +942,7 @@ static ssize_t show_io_db(struct device *dev,
965 struct resource_map *p; 942 struct resource_map *p;
966 ssize_t ret = 0; 943 ssize_t ret = 0;
967 944
968 mutex_lock(&rsrc_mutex); 945 mutex_lock(&s->ops_mutex);
969 data = s->resource_data; 946 data = s->resource_data;
970 947
971 for (p = data->io_db.next; p != &data->io_db; p = p->next) { 948 for (p = data->io_db.next; p != &data->io_db; p = p->next) {
@@ -977,7 +954,7 @@ static ssize_t show_io_db(struct device *dev,
977 ((unsigned long) p->base + p->num - 1)); 954 ((unsigned long) p->base + p->num - 1));
978 } 955 }
979 956
980 mutex_unlock(&rsrc_mutex); 957 mutex_unlock(&s->ops_mutex);
981 return ret; 958 return ret;
982} 959}
983 960
@@ -1005,9 +982,11 @@ static ssize_t store_io_db(struct device *dev,
1005 if (end_addr < start_addr) 982 if (end_addr < start_addr)
1006 return -EINVAL; 983 return -EINVAL;
1007 984
985 mutex_lock(&s->ops_mutex);
1008 ret = adjust_io(s, add, start_addr, end_addr); 986 ret = adjust_io(s, add, start_addr, end_addr);
1009 if (!ret) 987 if (!ret)
1010 s->resource_setup_new = 1; 988 s->resource_setup_new = 1;
989 mutex_unlock(&s->ops_mutex);
1011 990
1012 return ret ? ret : count; 991 return ret ? ret : count;
1013} 992}
@@ -1021,7 +1000,7 @@ static ssize_t show_mem_db(struct device *dev,
1021 struct resource_map *p; 1000 struct resource_map *p;
1022 ssize_t ret = 0; 1001 ssize_t ret = 0;
1023 1002
1024 mutex_lock(&rsrc_mutex); 1003 mutex_lock(&s->ops_mutex);
1025 data = s->resource_data; 1004 data = s->resource_data;
1026 1005
1027 for (p = data->mem_db.next; p != &data->mem_db; p = p->next) { 1006 for (p = data->mem_db.next; p != &data->mem_db; p = p->next) {
@@ -1033,7 +1012,7 @@ static ssize_t show_mem_db(struct device *dev,
1033 ((unsigned long) p->base + p->num - 1)); 1012 ((unsigned long) p->base + p->num - 1));
1034 } 1013 }
1035 1014
1036 mutex_unlock(&rsrc_mutex); 1015 mutex_unlock(&s->ops_mutex);
1037 return ret; 1016 return ret;
1038} 1017}
1039 1018
@@ -1061,9 +1040,11 @@ static ssize_t store_mem_db(struct device *dev,
1061 if (end_addr < start_addr) 1040 if (end_addr < start_addr)
1062 return -EINVAL; 1041 return -EINVAL;
1063 1042
1043 mutex_lock(&s->ops_mutex);
1064 ret = adjust_memory(s, add, start_addr, end_addr); 1044 ret = adjust_memory(s, add, start_addr, end_addr);
1065 if (!ret) 1045 if (!ret)
1066 s->resource_setup_new = 1; 1046 s->resource_setup_new = 1;
1047 mutex_unlock(&s->ops_mutex);
1067 1048
1068 return ret ? ret : count; 1049 return ret ? ret : count;
1069} 1050}