floppy drives and ACPI

Matthew Dillon dillon at apollo.backplane.com
Tue Oct 12 18:42:35 PDT 2004


    Ok, people whos floppies aren't recognized, please try this patch.

    I did not adopt the FreeBSD work, it was such a huge hack that there's
    no point even trying to decipher the mess.  Instead I rolled a routine
    to allocate all available resources of a particular type and return
    information about what it did, and then check that against reasonable
    values.

						-Matt

Index: dev/disk/fd/fd.c
===================================================================
RCS file: /cvs/src/sys/dev/disk/fd/fd.c,v
retrieving revision 1.19
diff -u -r1.19 fd.c
--- dev/disk/fd/fd.c	19 Sep 2004 00:36:37 -0000	1.19
+++ dev/disk/fd/fd.c	13 Oct 2004 01:24:50 -0000
@@ -313,7 +313,7 @@
 static void
 fdctl_wr_isa(fdc_p fdc, u_int8_t v)
 {
-	bus_space_write_1(fdc->ctlt, fdc->ctlh, 0, v);
+	bus_space_write_1(fdc->portt, fdc->porth, 7 + fdc->port_off, v);
 }
 
 #if 0
@@ -537,84 +537,107 @@
 {
 	device_t dev;
 	int ispnp, ispcmcia;
+	u_long range_start;
+	u_long range_size;
+	u_long actual_size;
+	int error;
 
 	dev = fdc->fdc_dev;
 	ispnp = (fdc->flags & FDC_ISPNP) != 0;
 	ispcmcia = (fdc->flags & FDC_ISPCMCIA) != 0;
-	fdc->rid_ioport = fdc->rid_irq = fdc->rid_drq = 0;
-	fdc->res_ioport = fdc->res_irq = fdc->res_drq = 0;
+	fdc->rid_irq = 0;
+	fdc->rid_drq = 0;
+	fdc->res_nioports = 0;
+	fdc->res_irq = NULL;
+	fdc->res_drq = NULL;
+	callout_init(&fdc->pseudointr_ch);
 
 	/*
 	 * On standard ISA, we don't just use an 8 port range
 	 * (e.g. 0x3f0-0x3f7) since that covers an IDE control
-	 * register at 0x3f6.
-	 *
-	 * Isn't PC hardware wonderful.
+	 * register at 0x3f6.  Isn't PC hardware wonderful?
 	 *
 	 * The Y-E Data PCMCIA FDC doesn't have this problem, it
 	 * uses the register with offset 6 for pseudo-DMA, and the
 	 * one with offset 7 as control register.
+	 *
+	 * Even worse, many BIOS makers improperly specify the ACPI
+	 * resource lists, sometimes starting at 0x3f2, sometimes only
+	 * listing a contiguous range of 2 bytes for the first (rid 0)
+	 * resource, and so forth.  Known combinations:
+	 *
+	 * 1:   0x3f0-0x3f5                     # very rare
+	 * 2:   0x3f0                           # hints -> 0x3f0-0x3f5,0x3f7
+	 * 3:   0x3f0-0x3f5,0x3f7               # Most common
+	 * 4:   0x3f2-0x3f5,0x3f7               # Second most common
+	 * 5:   0x3f2-0x3f5                     # implies 0x3f7 too.
+	 * 6:   0x3f2-0x3f3,0x3f4-0x3f5,0x3f7   # becoming common
+	 * 7:   0x3f2-0x3f3,0x3f4-0x3f5         # rare
+	 * 8:   0x3f0-0x3f1,0x3f2-0x3f3,0x3f4-0x3f5,0x3f7
+	 * 9:   0x3f0-0x3f3,0x3f4-0x3f5,0x3f7
+	 *
+	 * To deal with this mess we use bus_alloc_all_resources() which
+	 * will allocate all necessary resources within the area mask and
+	 * return the overall start and size, and the resource associated with
+	 * the lowest address found.
+	 *
+	 * For floppy contorllers we just assume that
 	 */
-	fdc->res_ioport = bus_alloc_resource(dev, SYS_RES_IOPORT,
-					     &fdc->rid_ioport, 0ul, ~0ul, 
-					     ispcmcia ? 8 : (ispnp ? 1 : 6),
-					     RF_ACTIVE);
-	if (fdc->res_ioport == 0) {
-		device_printf(dev, "cannot reserve I/O port range\n");
+	error = bus_alloc_all_resources(dev, SYS_RES_IOPORT, RF_ACTIVE,
+					fdc->res_ioports, FDC_MAX_IORESOURCES,
+					&fdc->res_nioports, 7, 
+					&range_start, &range_size, 
+					&actual_size, &fdc->res_ioport);
+	printf("floppy: nioresources %d range %04lx-%04lx (%ld)\n",
+		fdc->res_nioports, range_start, range_start + range_size,
+		actual_size);
+
+	if (error) {
+		device_printf(dev, "Has no conforming I/O port range\n");
 		return ENXIO;
 	}
-	fdc->portt = rman_get_bustag(fdc->res_ioport);
-	fdc->porth = rman_get_bushandle(fdc->res_ioport);
-
-	if (!ispcmcia) {
-		/*
-		 * Some BIOSen report the device at 0x3f2-0x3f5,0x3f7
-		 * and some at 0x3f0-0x3f5,0x3f7. We detect the former
-		 * by checking the size and adjust the port address
-		 * accordingly.
-		 */
-		if (bus_get_resource_count(dev, SYS_RES_IOPORT, 0) == 4)
-			fdc->port_off = -2;
 
+	/*
+	 * Check the masked start port offset for the 0x3f2 case.
+	 */
+	switch(range_start & 7) {
+	case 0:
+		fdc->port_off = 0;
+		break;
+	case 2:
 		/*
-		 * Register the control port range as rid 1 if it
-		 * isn't there already. Most PnP BIOSen will have
-		 * already done this but non-PnP configurations don't.
-		 *
-		 * And some (!!) report 0x3f2-0x3f5 and completely
-		 * leave out the control register!  It seems that some
-		 * non-antique controller chips have a different
-		 * method of programming the transfer speed which
-		 * doesn't require the control register, but it's
-		 * mighty bogus as the chip still responds to the
-		 * address for the control register.
+		 * BIOS resource list bug, started at 0x3f2 instead of 0x3f0.
 		 */
-		if (bus_get_resource_count(dev, SYS_RES_IOPORT, 1) == 0) {
-			u_long ctlstart;
-
-			/* Find the control port, usually 0x3f7 */
-			ctlstart = rman_get_start(fdc->res_ioport) +
-				fdc->port_off + 7;
-
-			bus_set_resource(dev, SYS_RES_IOPORT, 1, ctlstart, 1);
-		}
+		device_printf(dev, "Warning: corrected illegal port range\n");
+		fdc->port_off = -2;
+		actual_size += 2;	/* fake it */
+		break;
+	default:
+		device_printf(dev, "Illegal I/O port range\n");
+		return (ENXIO);
+	}
 
-		/*
-		 * Now (finally!) allocate the control port.
-		 */
-		fdc->rid_ctl = 1;
-		fdc->res_ctl = bus_alloc_resource(dev, SYS_RES_IOPORT,
-						  &fdc->rid_ctl,
-						  0ul, ~0ul, 1, RF_ACTIVE);
-		if (fdc->res_ctl == 0) {
-			device_printf(dev,
-				      "cannot reserve control I/O port range\n");
-			return ENXIO;
-		}
-		fdc->ctlt = rman_get_bustag(fdc->res_ctl);
-		fdc->ctlh = rman_get_bushandle(fdc->res_ctl);
+	/*
+	 * The actual size ought to be 0, 7, or 8.  It will be 0 if generated
+	 * from an ISA hint, 7 normally (the range doesn't include 0x3f6 which
+	 * is an IDE control register), or 8, which is usually a PCMCIA
+	 * card that doesn't have an IDE control register in the middle of
+	 * the floppy controller.
+	 *
+	 * Degenerate ISA hints will have a size of 1
+	 */
+	printf("actual size: %d\n", actual_size);
+	if (actual_size != 0 && actual_size < 7) {
+		device_printf(dev, "Illegal I/O port range\n");
+		return (ENXIO);
 	}
 
+	fdc->portt = rman_get_bustag(fdc->res_ioport);
+	fdc->porth = rman_get_bushandle(fdc->res_ioport);
+
+	/*
+	 * Move on to other resources
+	 */
 	fdc->res_irq = bus_alloc_resource(dev, SYS_RES_IRQ,
 					  &fdc->rid_irq, 0ul, ~0ul, 1, 
 					  RF_ACTIVE);
@@ -642,6 +665,7 @@
 {
 	device_t dev;
 
+	callout_stop(&fdc->pseudointr_ch);
 	dev = fdc->fdc_dev;
 	if (fdc->res_irq != 0) {
 		bus_deactivate_resource(dev, SYS_RES_IRQ, fdc->rid_irq,
@@ -649,17 +673,14 @@
 		bus_release_resource(dev, SYS_RES_IRQ, fdc->rid_irq,
 				     fdc->res_irq);
 	}
-	if (fdc->res_ctl != 0) {
-		bus_deactivate_resource(dev, SYS_RES_IOPORT, fdc->rid_ctl,
-					fdc->res_ctl);
-		bus_release_resource(dev, SYS_RES_IOPORT, fdc->rid_ctl,
-				     fdc->res_ctl);
-	}
-	if (fdc->res_ioport != 0) {
-		bus_deactivate_resource(dev, SYS_RES_IOPORT, fdc->rid_ioport,
-					fdc->res_ioport);
-		bus_release_resource(dev, SYS_RES_IOPORT, fdc->rid_ioport,
-				     fdc->res_ioport);
+
+	if (fdc->res_nioports) {
+		bus_deactivate_all_resources(dev, SYS_RES_IOPORT,
+					fdc->res_ioports, fdc->res_nioports);
+		bus_release_all_resources(dev, SYS_RES_IOPORT,
+					fdc->res_ioports, &fdc->res_nioports);
+		fdc->res_nioports = 0;
+		fdc->res_ioport = NULL;
 	}
 	if (fdc->res_drq != 0) {
 		bus_deactivate_resource(dev, SYS_RES_DRQ, fdc->rid_drq,
Index: dev/disk/fd/fdc.h
===================================================================
RCS file: /cvs/src/sys/dev/disk/fd/fdc.h,v
retrieving revision 1.4
diff -u -r1.4 fdc.h
--- dev/disk/fd/fdc.h	19 Sep 2004 00:36:37 -0000	1.4
+++ dev/disk/fd/fdc.h	13 Oct 2004 01:05:51 -0000
@@ -75,8 +75,15 @@
 	struct	resource *res_irq, *res_drq;
 	int	rid_ioport, rid_irq, rid_drq;
 #else
-	struct	resource *res_ioport, *res_ctl, *res_irq, *res_drq;
-	int	rid_ioport, rid_ctl, rid_irq, rid_drq;
+#define FDC_MAX_IORESOURCES	8
+	struct	resource_info res_ioports[FDC_MAX_IORESOURCES];
+	struct	resource *res_ioport;	/* lowest port from ioports[]*/
+	struct	resource *res_irq;
+	struct	resource *res_drq;
+	int	res_nioports;
+	int	rid_iostart;
+	int	rid_irq;
+	int	rid_drq;
 #endif
 	int	port_off;
 	bus_space_tag_t portt;
@@ -86,9 +93,6 @@
 	bus_space_handle_t	sc_fdsioh;
 	bus_space_tag_t		sc_fdemsiot;
 	bus_space_handle_t	sc_fdemsioh;
-#else
-	bus_space_tag_t ctlt;
-	bus_space_handle_t ctlh;
 #endif
 	void	*fdc_intr;
 	struct	device *fdc_dev;
Index: kern/subr_bus.c
===================================================================
RCS file: /cvs/src/sys/kern/subr_bus.c,v
retrieving revision 1.21
diff -u -r1.21 subr_bus.c
--- kern/subr_bus.c	8 Jul 2004 12:43:32 -0000	1.21
+++ kern/subr_bus.c	13 Oct 2004 01:16:22 -0000
@@ -2094,6 +2094,127 @@
 				  count, flags));
 }
 
+/*
+ * This routine allocates all resources which fall within the area mask.
+ * The first address retrieved determines the I/O area.  Resources are
+ * stuffed in the provided array (note that array entry 0 is left NULL)
+ *
+ * This is a real mess because the bus_*() infrastructure is a mess.
+ */
+int
+bus_alloc_all_resources(device_t dev, int type, u_int flags, 
+			struct resource_info *res_ary, int maxcount,
+			int *count, u_long area_mask,
+			u_long *range_start, u_long *range_size,
+			u_long *actual_size, struct resource **res_lo)
+{
+	struct resource *scan_res;
+	u_long	scan_start;
+	u_long	scan_size;
+	u_long	range_area;
+	int error;
+	int rid;
+	int i;
+
+	*res_lo = NULL;
+	*range_start = 0;
+	*range_size = 0;
+	*actual_size = 0;
+	*count = 0;
+
+	if (maxcount == 0)
+		return (EINVAL);
+
+	*range_start = 0;
+	*range_size = 0;
+	range_area = 0;
+	error = 0;
+
+	for (rid = 0; *count < maxcount; ++*count, ++rid) {
+		scan_res = bus_alloc_resource(dev, type, &rid, 0ul, ~0ul,
+						0, flags);
+		if (scan_res == NULL) {
+			if (*count == 0)
+				error = ENXIO;
+			break;
+		}
+		if (rid < 0 || rid >= maxcount) {
+			device_printf(dev, "bus_alloc_all_resources(): rid %d is out of range, cannot continue\n", rid);
+			bus_release_resource(dev, type, rid, scan_res);
+			error = ENXIO;
+			break;
+		}
+		scan_start = rman_get_start(scan_res);
+		scan_size = rman_get_size(scan_res);
+
+		/*
+		 * Initialize our initial range start in the first loop.
+		 */
+		if (*range_size == 0) {
+			*range_start = scan_start;
+			range_area = scan_start & ~area_mask;
+		}
+
+		/*
+		 * If we are beyond the requested area mask, stop.
+		 */
+		if (range_area != (scan_start & ~area_mask)) {
+			bus_release_resource(dev, type, rid, scan_res);
+			break;
+		}
+
+		/*
+		 * Update the resource array
+		 */
+		res_ary[*count].ri_res = scan_res;
+		res_ary[*count].ri_rid = rid;
+
+		/*
+		 * Clip the resource to the requested area mask and update
+		 * return parameters.
+		 */
+		if (range_area != ((scan_start + scan_size) & ~area_mask)) {
+			scan_size = ((scan_start + area_mask) & ~area_mask) -
+					scan_start;
+		}
+		if (*range_start > scan_start) {
+			*range_size += *range_start - scan_start;
+			*range_start = scan_start;
+		}
+		if (*range_start + *range_size < scan_start + scan_size) {
+			*range_size += (scan_start + scan_size) -
+					(*range_start + *range_size);
+		}
+		if (scan_start <= *range_start)
+			*res_lo = scan_res;
+		*actual_size += scan_size;
+	}
+
+	if (error) {
+		for (i = *count - 1; i >= 0; --i)
+			bus_release_resource(dev, type, res_ary[i].ri_rid,
+					     res_ary[i].ri_res);
+		*count = 0;
+		*res_lo = NULL;
+	}
+	return (error);
+}
+
+void
+bus_release_all_resources(device_t dev, int type,
+			  struct resource_info *res_ary, int *count)
+{
+	int i;
+
+	for (i = *count - 1; i >= 0; --i) {
+		bus_release_resource(dev, type,
+			res_ary[i].ri_rid, res_ary[i].ri_res);
+		res_ary[i].ri_rid = 0;
+		res_ary[i].ri_res = NULL;
+	}
+	*count = 0;
+}
+
 int
 bus_activate_resource(device_t dev, int type, int rid, struct resource *r)
 {
@@ -2111,6 +2232,24 @@
 }
 
 int
+bus_deactivate_all_resources(device_t dev, int type, 
+                                struct resource_info *res_ary, int count)
+{
+	int cumulative_error;
+	int error;
+
+	cumulative_error = 0;
+	for (--count; count >= 0; --count) {
+		error = bus_deactivate_resource(dev, type,
+				res_ary[count].ri_rid, res_ary[count].ri_res);
+		if (error)
+			cumulative_error = error;
+	}
+	return(cumulative_error);
+}
+
+
+int
 bus_release_resource(device_t dev, int type, int rid, struct resource *r)
 {
 	if (dev->parent == 0)
Index: sys/bus.h
===================================================================
RCS file: /cvs/src/sys/sys/bus.h,v
retrieving revision 1.12
diff -u -r1.12 bus.h
--- sys/bus.h	5 May 2004 16:57:11 -0000	1.12
+++ sys/bus.h	13 Oct 2004 01:15:34 -0000
@@ -96,6 +96,7 @@
  * for their child devices.
  */
 struct	resource;
+struct	resource_info;
 
 struct resource_list_entry {
     SLIST_ENTRY(resource_list_entry) link;
@@ -236,10 +237,19 @@
 struct	resource *bus_alloc_resource(device_t dev, int type, int *rid,
 				     u_long start, u_long end, u_long count,
 				     u_int flags);
+int	bus_alloc_all_resources(device_t dev, int type, u_int flags,
+				struct resource_info *res_ary, int maxcount,
+				int *count, u_long area_mask,
+				u_long *range_start, u_long *range_size,
+				u_long *actual_size, struct resource **res_lo);
+void	bus_release_all_resources(device_t dev, int type,
+				  struct resource_info *res_ary, int *count);
 int	bus_activate_resource(device_t dev, int type, int rid, 
 			      struct resource *r);
 int	bus_deactivate_resource(device_t dev, int type, int rid,
 				struct resource *r);
+int	bus_deactivate_all_resources(device_t dev, int type, 
+				struct resource_info *res_ary, int count);
 int	bus_release_resource(device_t dev, int type, int rid, 
 			     struct resource *r);
 int	bus_setup_intr(device_t dev, struct resource *r, int flags,
Index: sys/rman.h
===================================================================
RCS file: /cvs/src/sys/sys/rman.h,v
retrieving revision 1.7
diff -u -r1.7 rman.h
--- sys/rman.h	1 Mar 2004 06:33:19 -0000	1.7
+++ sys/rman.h	13 Oct 2004 01:05:14 -0000
@@ -61,6 +61,11 @@
 	struct	rman *r_rm;	/* resource manager from whence this came */
 };
 
+struct  resource_info {
+	struct resource		*ri_res;
+	int			ri_rid;
+};
+
 #define	RF_ALLOCATED	0x0001	/* resource has been reserved */
 #define	RF_ACTIVE	0x0002	/* resource allocation has been activated */
 #define	RF_SHAREABLE	0x0004	/* resource permits contemporaneous sharing */





More information about the Bugs mailing list