<html><head><meta name="color-scheme" content="light dark"></head><body><pre style="word-wrap: break-word; white-space: pre-wrap;">
From: Dominik Brodowski &lt;linux@dominikbrodowski.net&gt;

Always rescan the devices upon echo'ing something to
available_resources_setup_done.  This is needed for proper "coldplug" support.

Signed-off-by: Dominik Brodowski &lt;linux@dominikbrodowski.net&gt;
Signed-off-by: Andrew Morton &lt;akpm@osdl.org&gt;
---

 drivers/pcmcia/cs_internal.h  |    3 +--
 drivers/pcmcia/ds.c           |   38 +++++++++++++++++++++++++++++++++++---
 drivers/pcmcia/socket_sysfs.c |   26 +++++++++++---------------
 3 files changed, 47 insertions(+), 20 deletions(-)

diff -puN drivers/pcmcia/cs_internal.h~pcmcia-rescan-bus-always-upon-echoing-into-setup_done drivers/pcmcia/cs_internal.h
--- 25/drivers/pcmcia/cs_internal.h~pcmcia-rescan-bus-always-upon-echoing-into-setup_done	2005-05-31 02:44:40.000000000 -0700
+++ 25-akpm/drivers/pcmcia/cs_internal.h	2005-05-31 02:44:40.000000000 -0700
@@ -159,8 +159,7 @@ int pccard_access_configuration_register
 struct pcmcia_callback{
 	struct module	*owner;
 	int		(*event) (struct pcmcia_socket *s, event_t event, int priority);
-	int		(*resources_done) (struct pcmcia_socket *s);
-	void		(*replace_cis) (void);
+	void		(*requery) (struct pcmcia_socket *s);
 };
 
 int pccard_register_pcmcia(struct pcmcia_socket *s, struct pcmcia_callback *c);
diff -puN drivers/pcmcia/ds.c~pcmcia-rescan-bus-always-upon-echoing-into-setup_done drivers/pcmcia/ds.c
--- 25/drivers/pcmcia/ds.c~pcmcia-rescan-bus-always-upon-echoing-into-setup_done	2005-05-31 02:44:40.000000000 -0700
+++ 25-akpm/drivers/pcmcia/ds.c	2005-05-31 02:44:40.000000000 -0700
@@ -717,9 +717,42 @@ static inline void pcmcia_add_pseudo_dev
 	return;
 }
 
-static void pcmcia_bus_rescan(void)
+static int pcmcia_requery(struct device *dev, void * _data)
 {
+	struct pcmcia_device *p_dev = to_pcmcia_dev(dev);
+	if (!p_dev-&gt;dev.driver)
+		pcmcia_device_query(p_dev);
+
+	return 0;
+}
+
+static void pcmcia_bus_rescan(struct pcmcia_socket *skt)
+{
+	int no_devices=0;
+	unsigned long flags;
+
 	/* must be called with skt_sem held */
+	spin_lock_irqsave(&amp;pcmcia_dev_list_lock, flags);
+	if (list_empty(&amp;skt-&gt;pcmcia-&gt;devices_list))
+		no_devices=1;
+	spin_unlock_irqrestore(&amp;pcmcia_dev_list_lock, flags);
+
+	/* if no devices were added for this socket yet because of
+	 * missing resource information or other trouble, we need to
+	 * do this now. */
+	if (no_devices) {
+		int ret = pcmcia_card_add(skt);
+		if (ret)
+			return;
+	}
+
+	/* some device information might have changed because of a CIS
+	 * update or because we can finally read it correctly... so
+	 * determine it again, overwriting old values if necessary. */
+	bus_for_each_dev(&amp;pcmcia_bus_type, NULL, NULL, pcmcia_requery);
+
+	/* we re-scan all devices, not just the ones connected to this
+	 * socket. This does not matter, though. */
 	bus_rescan_devices(&amp;pcmcia_bus_type);
 }
 
@@ -1861,8 +1894,7 @@ static int __devinit pcmcia_bus_add_sock
 	/* Set up hotline to Card Services */
 	s-&gt;callback.owner = THIS_MODULE;
 	s-&gt;callback.event = &amp;ds_event;
-	s-&gt;callback.resources_done = &amp;pcmcia_card_add;
-	s-&gt;callback.replace_cis = &amp;pcmcia_bus_rescan;
+	s-&gt;callback.requery = &amp;pcmcia_bus_rescan;
 	socket-&gt;pcmcia = s;
 
 	ret = pccard_register_pcmcia(socket, &amp;s-&gt;callback);
diff -puN drivers/pcmcia/socket_sysfs.c~pcmcia-rescan-bus-always-upon-echoing-into-setup_done drivers/pcmcia/socket_sysfs.c
--- 25/drivers/pcmcia/socket_sysfs.c~pcmcia-rescan-bus-always-upon-echoing-into-setup_done	2005-05-31 02:44:40.000000000 -0700
+++ 25-akpm/drivers/pcmcia/socket_sysfs.c	2005-05-31 02:44:40.000000000 -0700
@@ -163,24 +163,20 @@ static ssize_t pccard_store_resource(str
 		return -EINVAL;
 
 	spin_lock_irqsave(&amp;s-&gt;lock, flags);
-	if (!s-&gt;resource_setup_done) {
+	if (!s-&gt;resource_setup_done)
 		s-&gt;resource_setup_done = 1;
-		spin_unlock_irqrestore(&amp;s-&gt;lock, flags);
+	spin_unlock_irqrestore(&amp;s-&gt;lock, flags);
 
-		down(&amp;s-&gt;skt_sem);
-		if ((s-&gt;callback) &amp;&amp;
-		    (s-&gt;state &amp; SOCKET_PRESENT) &amp;&amp;
-		    !(s-&gt;state &amp; SOCKET_CARDBUS)) {
-			if (try_module_get(s-&gt;callback-&gt;owner)) {
-				s-&gt;callback-&gt;resources_done(s);
-				module_put(s-&gt;callback-&gt;owner);
-			}
+	down(&amp;s-&gt;skt_sem);
+	if ((s-&gt;callback) &amp;&amp;
+	    (s-&gt;state &amp; SOCKET_PRESENT) &amp;&amp;
+	    !(s-&gt;state &amp; SOCKET_CARDBUS)) {
+		if (try_module_get(s-&gt;callback-&gt;owner)) {
+			s-&gt;callback-&gt;requery(s);
+			module_put(s-&gt;callback-&gt;owner);
 		}
-		up(&amp;s-&gt;skt_sem);
-
-		return count;
 	}
-	spin_unlock_irqrestore(&amp;s-&gt;lock, flags);
+	up(&amp;s-&gt;skt_sem);
 
 	return count;
 }
@@ -315,7 +311,7 @@ static ssize_t pccard_store_cis(struct k
 		if ((s-&gt;callback) &amp;&amp; (s-&gt;state &amp; SOCKET_PRESENT) &amp;&amp;
 		    !(s-&gt;state &amp; SOCKET_CARDBUS)) {
 			if (try_module_get(s-&gt;callback-&gt;owner)) {
-				s-&gt;callback-&gt;replace_cis();
+				s-&gt;callback-&gt;requery(s);
 				module_put(s-&gt;callback-&gt;owner);
 			}
 		}
_
</pre></body></html>