From: Laurent Vivier <[EMAIL PROTECTED]>

This patch is a port of the driver I wrote for kernel 2.2.

It allows to read data from a floppy, but not to write to, and to eject the
floppy (useful on our Mac without eject button).

I have tested it on a Quadra 800, but it is supposed to work on:

II, IIci, IIsi, IIvx, IIx, IIcx, SE/30,
PowerBook 100, PowerBook 140, PowerBook 145, PowerBook 160,
PowerBook 165, PowerBook 165c, PowerBook 170, PowerBook 180,
PowerBook 180c, PowerBook 190, PowerBook 190cs, PowerBook 500,
Performa 460, Performa 550, LC II, LC III, LC 520, Color Classic,
Color Classic II, ClassicII,
Quadra 700, Quadra 800, Quadra 650, Quadra 605, Quadra 610,
Centris 610, Quadra 630, Performa 580, LC 475, LC 575,

Signed-off-by: Laurent Vivier <[EMAIL PROTECTED]>
---
 arch/m68k/mac/Makefile   |    2 
 arch/m68k/mac/config.c   |    2 
 arch/m68k/mac/swim.c     |  111 +++++
 arch/m68k/mac/via.c      |    1 
 drivers/block/Kconfig    |   16 
 drivers/block/Makefile   |    3 
 drivers/block/swim.c     |  886 +++++++++++++++++++++++++++++++++++++++++++++++
 drivers/block/swim_asm.S |  295 +++++++++++++++
 8 files changed, 1315 insertions(+), 1 deletion(-)

Index: linux-2.6/arch/m68k/mac/Makefile
===================================================================
--- linux-2.6.orig/arch/m68k/mac/Makefile       2008-11-01 06:13:53.000000000 
+0100
+++ linux-2.6/arch/m68k/mac/Makefile    2008-11-01 06:14:39.000000000 +0100
@@ -3,4 +3,4 @@
 #
 
 obj-y          := config.o macints.o iop.o via.o oss.o psc.o \
-                       baboon.o macboing.o debug.o misc.o
+                       baboon.o macboing.o debug.o misc.o swim.o
Index: linux-2.6/arch/m68k/mac/config.c
===================================================================
--- linux-2.6.orig/arch/m68k/mac/config.c       2008-11-01 06:13:53.000000000 
+0100
+++ linux-2.6/arch/m68k/mac/config.c    2008-11-01 06:14:39.000000000 +0100
@@ -70,6 +70,7 @@
 extern void oss_init(void);
 extern void psc_init(void);
 extern void baboon_init(void);
+extern void swim_init(void);
 
 extern void mac_mksound(unsigned int, unsigned int);
 
@@ -815,6 +816,7 @@
        oss_init();
        psc_init();
        baboon_init();
+       swim_init();
 }
 
 static void __init mac_report_hardware(void)
Index: linux-2.6/arch/m68k/mac/swim.c
===================================================================
--- /dev/null   1970-01-01 00:00:00.000000000 +0000
+++ linux-2.6/arch/m68k/mac/swim.c      2008-11-01 06:14:39.000000000 +0100
@@ -0,0 +1,111 @@
+/*
+ * Driver for SWIM (Sander. Woz Integrated Machine) floppy controller
+ *
+ * Copyright (C) 2004,2008 Laurent Vivier <[EMAIL PROTECTED]>
+ *
+ * based on netBSD IWM driver (c) 1997, 1998 Hauke Fath.
+ * based  on Alastair Bridgewater SWIM analysis, 2001
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version
+ * 2 of the License, or (at your option) any later version.
+ *
+ * 2004-09-02 (lv) - Initial implementation
+ * 2008-10-30 (lv) - Port to kernel 2.6
+ */
+
+#include <linux/types.h>
+#include <linux/kernel.h>
+#include <linux/mm.h>
+#include <linux/init.h>
+#include <linux/module.h>
+
+#include <asm/macintosh.h>
+#include <asm/mac_via.h>
+
+volatile __u8 *SWIMBase;
+EXPORT_SYMBOL(SWIMBase);
+
+#define writePhase     *(SWIMBase + 0x0800)
+#define readPhase      *(SWIMBase + 0x1800)
+
+/*
+ * According to IWM netBSD driver, there are four kinds of SWIM:
+ *
+ * - QUADRA, QUADRA2, P580 -> SWIM base address is VIA1 + 0x1E000;
+ * - II, PB, LC                   -> SWIM base address is VIA1 + 0x16000;
+ * - IIfx, Q900, Q950     -> managed by IOP driver
+ * - AV                           -> not managed
+ *
+ */
+
+void __init swim_init(void)
+{
+       switch(macintosh_config->ident)
+       {
+       case MAC_MODEL_Q700:
+       case MAC_MODEL_Q800:
+       case MAC_MODEL_Q650:
+       case MAC_MODEL_Q605:
+       case MAC_MODEL_Q605_ACC:
+       case MAC_MODEL_Q610:
+       case MAC_MODEL_Q630:
+       case MAC_MODEL_P475:
+       case MAC_MODEL_P475F:
+       case MAC_MODEL_P575:
+       case MAC_MODEL_P588:
+               SWIMBase =  (__u8*)(VIA1_BASE + 0x1E000);
+               break;
+       case MAC_MODEL_II:
+       case MAC_MODEL_IIX:
+       case MAC_MODEL_IICX:
+       case MAC_MODEL_SE30:
+       case MAC_MODEL_PB140:
+       case MAC_MODEL_PB145:
+       case MAC_MODEL_PB160:
+       case MAC_MODEL_PB165:
+       case MAC_MODEL_PB165C:
+       case MAC_MODEL_PB170:
+       case MAC_MODEL_PB180:
+       case MAC_MODEL_PB180C:
+       case MAC_MODEL_PB190:
+       case MAC_MODEL_PB520:
+       case MAC_MODEL_PB150:
+       case MAC_MODEL_PB210:
+       case MAC_MODEL_PB230:
+       case MAC_MODEL_PB250:
+       case MAC_MODEL_PB270C:
+       case MAC_MODEL_PB280:
+       case MAC_MODEL_PB280C:
+       case MAC_MODEL_IICI:
+       case MAC_MODEL_IISI:
+       case MAC_MODEL_IIVI:
+       case MAC_MODEL_IIVX:
+       case MAC_MODEL_P600:
+       case MAC_MODEL_P460:
+       case MAC_MODEL_P550:
+       case MAC_MODEL_TV:
+       case MAC_MODEL_LCII:
+       case MAC_MODEL_LCIII:
+       case MAC_MODEL_P520:
+       case MAC_MODEL_CLII:
+       case MAC_MODEL_CCL:
+               SWIMBase =  (__u8*)(VIA1_BASE + 0x16000);
+               break;
+       case MAC_MODEL_IIFX:
+       case MAC_MODEL_Q900:
+       case MAC_MODEL_Q950:
+               SWIMBase = NULL;
+               break;
+       default:
+               SWIMBase = NULL;
+               printk("SWIM: unknown Macintosh: report to maintainer !\n");
+               break;
+       }
+
+       if (SWIMBase == NULL)
+               return;
+
+       printk("SWIM floppy controller base at 0x%p\n", SWIMBase);
+}
Index: linux-2.6/arch/m68k/mac/via.c
===================================================================
--- linux-2.6.orig/arch/m68k/mac/via.c  2008-11-01 06:13:53.000000000 +0100
+++ linux-2.6/arch/m68k/mac/via.c       2008-11-01 06:14:39.000000000 +0100
@@ -36,6 +36,7 @@
 #include <asm/mac_psc.h>
 
 volatile __u8 *via1, *via2;
+EXPORT_SYMBOL(via1);
 int rbv_present;
 int via_alt_mapping;
 EXPORT_SYMBOL(via_alt_mapping);
Index: linux-2.6/drivers/block/Kconfig
===================================================================
--- linux-2.6.orig/drivers/block/Kconfig        2008-11-01 06:13:53.000000000 
+0100
+++ linux-2.6/drivers/block/Kconfig     2008-11-01 06:14:39.000000000 +0100
@@ -44,6 +44,22 @@
          If you have a SWIM-3 (Super Woz Integrated Machine 3; from Apple)
          floppy controller, say Y here. Most commonly found in PowerMacs.
 
+config BLK_DEV_SWIM
+       tristate "Support for SWIM Macintosh floppy"
+       depends on M68K && MAC
+       help
+           You should select this option, if you want floppy support and
+           you have one of following macintoshes:
+
+           II, IIci, IIsi, IIvx, IIx, IIcx, SE/30,
+           PowerBook 100, PowerBook 140, PowerBook 145, PowerBook 160,
+           PowerBook 165, PowerBook 165c, PowerBook 170, PowerBook 180,
+           PowerBook 180c, PowerBook 190, PowerBook 190cs, PowerBook 500,
+           Performa 460, Performa 550, LC II, LC III, LC 520, Color Classic,
+           Color Classic II, ClassicII,
+           Quadra 700, Quadra 800, Quadra 650, Quadra 605, Quadra 610,
+           Centris 610, Quadra 630, Performa 580, LC 475, LC 575,
+
 config AMIGA_Z2RAM
        tristate "Amiga Zorro II ramdisk support"
        depends on ZORRO
Index: linux-2.6/drivers/block/Makefile
===================================================================
--- linux-2.6.orig/drivers/block/Makefile       2008-11-01 06:13:53.000000000 
+0100
+++ linux-2.6/drivers/block/Makefile    2008-11-01 06:14:39.000000000 +0100
@@ -6,6 +6,7 @@
 # 
 
 obj-$(CONFIG_MAC_FLOPPY)       += swim3.o
+obj-$(CONFIG_BLK_DEV_SWIM)     += swim_mod.o
 obj-$(CONFIG_BLK_DEV_FD)       += floppy.o
 obj-$(CONFIG_AMIGA_FLOPPY)     += amiflop.o
 obj-$(CONFIG_PS3_DISK)         += ps3disk.o
@@ -32,3 +33,5 @@
 obj-$(CONFIG_BLK_DEV_HD)       += hd.o
 
 obj-$(CONFIG_XEN_BLKDEV_FRONTEND)      += xen-blkfront.o
+
+swim_mod-objs  := swim.o swim_asm.o
Index: linux-2.6/drivers/block/swim.c
===================================================================
--- /dev/null   1970-01-01 00:00:00.000000000 +0000
+++ linux-2.6/drivers/block/swim.c      2008-11-01 10:36:05.000000000 +0100
@@ -0,0 +1,886 @@
+/*
+ * Driver for SWIM (Sander. Woz Integrated Machine) floppy controller
+ *
+ * Copyright (C) 2004,2008 Laurent Vivier <[EMAIL PROTECTED]>
+ *
+ * based on Alastair Bridgewater SWIM analysis, 2001
+ * based on SWIM3 driver (c) Paul Mackerras, 1996
+ * based on netBSD IWM driver (c) 1997, 1998 Hauke Fath.
+ *
+ * Supported Macintoshes:
+ *
+ *   II, IIci, IIsi, IIvx, IIx, IIcx, SE/30,
+ *   PowerBook 100, PowerBook 140, PowerBook 145, PowerBook 160,
+ *   PowerBook 165, PowerBook 165c, PowerBook 170, PowerBook 180,
+ *   PowerBook 180c, PowerBook 190, PowerBook 190cs, PowerBook 500,
+ *   Performa 460, Performa 550, LC II, LC III, LC 520, Color Classic,
+ *   Color Classic II, ClassicII,
+ *   Quadra 700, Quadra 800, Quadra 650, Quadra 605, Quadra 610,
+ *   Centris 610, Quadra 630, Performa 580, LC 475, LC 575,
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version
+ * 2 of the License, or (at your option) any later version.
+ *
+ * 2004-08-21 (lv) - Initial implementation
+ * 2008-10-30 (lv) - Port to 2.6
+ */
+
+#include <linux/module.h>
+#include <linux/fd.h>
+#include <linux/blkdev.h>
+#include <linux/hdreg.h>
+#include <linux/kernel.h>
+#include <linux/delay.h>
+
+#include <asm/mac_via.h>
+
+static char *base __devinitdata = 0;
+module_param(base, charp, 0444);
+MODULE_PARM_DESC(base, "Base address of SWIM chip.");
+
+struct sector_header {
+       unsigned char side;
+       unsigned char track;
+       unsigned char sector;
+       unsigned char size;
+       unsigned char crc0;
+       unsigned char crc1;
+} __attribute__((packed));
+
+extern int swim_mode(int enable);
+extern int swim_read_sector_header(struct sector_header* header);
+extern int swim_read_sector_data(unsigned char *data);
+
+#define DRIVER_VERSION "Version 0.2 (2008-10-30)"
+
+#define REG(x) volatile unsigned char x, x ## _pad[0x200 - 1];
+
+struct swim {
+       REG(write_data)
+       REG(write_mark)
+       REG(write_CRC)
+       REG(write_parameter)
+       REG(write_phase)
+       REG(write_setup)
+       REG(write_mode0)
+       REG(write_mode1)
+
+       REG(read_data)
+       REG(read_mark)
+       REG(read_error)
+       REG(read_parameter)
+       REG(read_phase)
+       REG(read_setup)
+       REG(read_status)
+       REG(read_handshake)
+} __attribute__((packed));
+
+extern struct swim *SWIMBase;
+
+#define swim_write(reg, v)     SWIMBase->write_##reg = v
+#define swim_read(reg)         SWIMBase->read_##reg
+
+/* bits in phase register */
+
+#define SEEK_POSITIVE  0x070
+#define SEEK_NEGATIVE  0x074
+#define STEP           0x071
+#define MOTOR_ON       0x072
+#define MOTOR_OFF      0x076
+#define INDEX          0x073
+#define EJECT          0x077
+#define SETMFM         0x171
+#define SETGCR         0x175
+
+#define RELAX          0x033
+#define LSTRB          0x008
+
+#define CA_MASK                0x077
+
+/* Select values for swim_select and swim_readbit */
+
+#define STEP_DIR       0x070
+#define STEPPING       0x071
+#define MOTOR_ON       0x072
+#define ELAX           0x073   /* also eject in progress */
+#define READ_DATA_0    0x074
+#define TWOMEG_DRIVE   0x075
+#define SINGLE_SIDED   0x076
+#define DRIVE_PRESENT  0x077
+#define DISK_IN                0x170
+#define WRITE_PROT     0x171
+#define TRACK_ZERO     0x172
+#define TACHO          0x173
+#define READ_DATA_1    0x174
+#define MFM_MODE       0x175
+#define SEEK_COMPLETE  0x176
+#define ONEMEG_MEDIA   0x177
+
+/* Bits in handshake register */
+
+#define MARK_BYTE      0x01
+#define CRC_ZERO       0x02
+#define RDDATA         0x04
+#define SENSE          0x08
+#define MOTEN          0x10
+#define ERROR          0x20
+#define DAT2BYTE       0x40
+#define DAT1BYTE       0x80
+
+/* bits in setup register */
+
+#define S_INV_WDATA    0x01
+#define S_3_5_SELECT   0x02
+#define S_GCR          0x04
+#define S_FCLK_DIV2    0x08
+#define S_ERROR_CORR   0x10
+#define S_IBM_DRIVE    0x20
+#define S_GCR_WRITE    0x40
+#define S_TIMEOUT      0x80
+
+/* bits in mode register */
+
+#define CLFIFO         0x01
+#define ENBL1          0x02
+#define ENBL2          0x04
+#define ACTION         0x08
+#define        WRITE_MODE      0x10
+#define        HEDSEL          0x20
+#define        MOTON           0x80
+
+
+/*----------------------------------------------------------------------------*/
+
+typedef enum {
+       INTERNAL_DRIVE = 0x02,
+       EXTERNAL_DRIVE = 0x04,
+} drive_location_t;
+
+typedef enum {
+       DD_MEDIA,
+       HD_MEDIA,
+} media_type_t;
+
+struct floppy_state {
+
+       /* physical properties */
+
+       drive_location_t location;      /* internal or external drive */
+       int              head_number;   /* single- or double-sided drive */
+
+       /* media */
+
+       int              disk_in;
+       int              ejected;
+       media_type_t     type;
+       int              write_protected;
+
+       int              total_secs;
+       int              secpercyl;
+       int              secpertrack;
+
+       /* in-use information */
+
+       int             track;
+       int             ref_count;
+
+       struct gendisk *disk;
+};
+
+typedef enum {
+       OFF,
+       ON,
+} motor_action_t;
+
+typedef enum {
+       LOWER_HEAD = 0,
+       UPPER_HEAD = 1,
+} head_t;
+
+static struct request_queue *floppy_queue;
+
+static DEFINE_SPINLOCK(swim_lock);
+
+#define FD_MAX_UNIT    2
+
+static struct floppy_state unit[FD_MAX_UNIT];
+static int floppy_count = 0;
+
+static inline void swim_select(int sel)
+{
+       swim_write(phase, RELAX);
+
+       if (sel & 0x100)
+               via1[vBufA] |= VIA1A_vHeadSel;
+       else
+               via1[vBufA] &= ~VIA1A_vHeadSel;
+
+       swim_write(phase, sel & CA_MASK);
+}
+
+static inline void swim_action(int action)
+{
+       swim_select(action);
+       udelay(1);
+       swim_write(phase, (LSTRB<<4) | LSTRB);
+       udelay(1);
+       swim_write(phase, (LSTRB<<4) | ((~LSTRB) & 0x0F));
+       udelay(1);
+       swim_write(phase, RELAX);
+}
+
+static inline int swim_readbit(int bit)
+{
+       int stat;
+
+       swim_select(bit);
+
+       udelay(10);
+
+       stat = swim_read(handshake);
+
+       swim_write(phase, RELAX);
+
+       return (stat & SENSE) == 0;
+}
+
+static inline void swim_drive(drive_location_t location)
+{
+       if (location == INTERNAL_DRIVE) {
+               swim_write(mode0, EXTERNAL_DRIVE);      /* clear drive 1 bit */
+               swim_write(mode1, INTERNAL_DRIVE);      /* set drive 0 bit */
+       } else if (location == EXTERNAL_DRIVE) {
+               swim_write(mode0, INTERNAL_DRIVE);      /* clear drive 0 bit */
+               swim_write(mode1, EXTERNAL_DRIVE);      /* set drive 1 bit */
+       }
+}
+
+static inline void swim_motor(motor_action_t action)
+{
+       if (action == ON) {
+               int i;
+
+               swim_action(MOTOR_ON);
+
+               for (i = 0; i < 2*HZ; i++) {
+                       if (swim_readbit(MOTOR_ON))
+                               break;
+                       current->state = TASK_INTERRUPTIBLE;
+                       schedule_timeout(1);
+               }
+       } else if (action == OFF)
+               swim_action(MOTOR_OFF);
+}
+
+static inline void swim_eject(void)
+{
+       int i;
+
+       swim_action(EJECT);
+
+       for (i = 0; i < 2*HZ; i++) {
+               if (swim_readbit(RELAX))
+                       break;
+               current->state = TASK_INTERRUPTIBLE;
+               schedule_timeout(1);
+       }
+}
+
+static inline void swim_head(head_t head)
+{
+       /* FIXME: IWM reads bits SEL, CA2, CA1 to wait drive ready... */
+
+       /* wait drive is ready */
+
+       if (head == UPPER_HEAD)
+               swim_select(READ_DATA_1);
+       else if (head == LOWER_HEAD)
+               swim_select(READ_DATA_0);
+}
+
+static inline int swim_step(void)
+{
+       int wait;
+
+       swim_action(STEP);
+
+       for (wait = 0; wait < 80; wait++) {
+
+               current->state = TASK_INTERRUPTIBLE;
+               schedule_timeout(1);
+
+               if (!swim_readbit(STEPPING))
+                       return 0;
+       }
+
+       return -1;
+}
+
+static inline int swim_track00(void)
+{
+       int try;
+
+       swim_motor(ON);
+
+       swim_action(SEEK_NEGATIVE);
+
+       for (try = 0; try < 100; try++) {
+
+               if (swim_readbit(TRACK_ZERO))
+                       break;
+
+               if (swim_step())
+                       return -1;
+       }
+
+       if (swim_readbit(TRACK_ZERO))
+               return 0;
+
+       return -1;
+}
+
+static inline int swim_seek(int step)
+{
+       if (step == 0)
+               return 0;
+
+       swim_motor(ON);
+
+       if (step < 0) {
+               swim_action(SEEK_NEGATIVE);
+               step = -step;
+       } else
+               swim_action(SEEK_POSITIVE);
+
+       for ( ; step > 0; step--) {
+               if (swim_step())
+                       return -1;
+       }
+
+       return 0;
+}
+
+static inline int swim_track(struct floppy_state *fs,  int track)
+{
+       int ret;
+
+       ret = swim_seek(track - fs->track);
+
+       if (ret == 0)
+               fs->track = track;
+       else {
+               swim_track00();
+               fs->track = 0;
+       }
+
+       return ret;
+}
+
+static int floppy_eject(struct floppy_state *fs)
+{
+       swim_drive(fs->location);
+       swim_motor(OFF);
+       swim_eject();
+
+       fs->disk_in = 0;
+       fs->ejected = 1;
+
+       return 0;
+}
+
+static inline int swim_read_sector(struct floppy_state *fs,
+                                  int side, int track,
+                                  int sector, unsigned char *buffer)
+{
+       unsigned long flags;
+       struct sector_header header;
+       int ret = -1;
+       short i;
+
+       swim_track(fs, track);
+
+       swim_write(mode1, MOTON);
+       swim_head(side);
+       swim_write(mode0, side);
+
+       local_save_flags(flags);
+       local_irq_disable();
+       for (i = 0; i < 20000; i++) {
+               ret = swim_read_sector_header(&header);
+               if ( !ret && (header.sector == sector) ) {
+
+                       /* found */
+
+                       ret = swim_read_sector_data(buffer);
+                       break;
+               }
+       }
+       swim_write(mode0, MOTON);
+       local_irq_restore(flags);
+
+       if ( (header.side != side)  || (header.track != track) ||
+            (header.sector != sector) )
+               return 0;
+
+       return ret;
+}
+
+static int floppy_read_sectors(struct floppy_state *fs,
+                              int req_sector, int sectors_nb,
+                              unsigned char* buffer)
+{
+       int ret;
+       int side;
+       int track;
+       int sector;
+       int i;
+       int try;
+
+       swim_drive(fs->location);
+       for (i = req_sector; i < req_sector + sectors_nb; i++) {
+               int x;
+               track = i / fs->secpercyl;
+               x = i % fs->secpercyl;
+               side = x / fs->secpertrack;
+               sector = x % fs->secpertrack + 1;
+
+               try = 5;
+               do {
+                       ret = swim_read_sector(fs, side, track, sector,
+                                               buffer);
+                       if (try-- == 0)
+                               return -1;
+               } while(ret != 512);
+
+               buffer += ret;
+       }
+
+       return 0;
+}
+
+static void redo_fd_request(void)
+{
+       struct request *req;
+       struct floppy_state *fs;
+
+       while((req = elv_next_request(floppy_queue))) {
+
+               fs = &unit[(long)req->rq_disk->private_data];
+               if (req->sector < 0 || req->sector >= fs->total_secs) {
+                       end_request(req, 0);
+                       continue;
+               }
+               if (req->current_nr_sectors == 0) {
+                       end_request(req, 1);
+                       continue;
+               }
+               if (!fs->disk_in) {
+                       end_request(req, 0);
+                       continue;
+               }
+               if (rq_data_dir(req) == WRITE) {
+                       if (fs->write_protected) {
+                                end_request(req, 0);
+                                continue;
+                        }
+               }
+               switch(rq_data_dir(req)) {
+               case WRITE:
+                       /* NOT IMPLEMENTED */
+                       end_request(req, 0);
+                       break;
+               case READ:
+                       if (floppy_read_sectors(fs, req->sector,
+                                               req->current_nr_sectors,
+                                               req->buffer)) {
+                               end_request(req, 0);
+                               continue;
+                       }
+                       req->nr_sectors -= req->current_nr_sectors;
+                       req->sector += req->current_nr_sectors;
+                       req->buffer += req->current_nr_sectors * 512;
+                       end_request(req, 1);
+                       break;
+               }
+       }
+}
+
+static void do_fd_request(struct request_queue * q)
+{
+       redo_fd_request();
+}
+
+static struct floppy_struct floppy_type[4] = {
+       {    0, 0,0, 0,0,0x00,0x00,0x00,0x00,NULL },    /*  0 no testing    */
+       {  720, 9,1,80,0,0x2A,0x02,0xDF,0x50,NULL },    /*  3 360KB SS 3.5" */
+       { 1440, 9,2,80,0,0x2A,0x02,0xDF,0x50,NULL },    /*  4 720KB 3.5"    */
+       { 2880,18,2,80,0,0x1B,0x00,0xCF,0x6C,NULL },    /*  7 1.44MB 3.5"   */
+};
+
+static int get_floppy_geometry(int drive, int type, struct floppy_struct **g)
+{
+       struct floppy_state *fs;
+       fs = &unit[drive];
+
+       if (drive > floppy_count)
+               return -ENODEV;
+
+       if (type >= ARRAY_SIZE(floppy_type))
+               return -EINVAL;
+
+       if (type)
+               *g = &floppy_type[type];
+       else if (fs->type == HD_MEDIA) /* High-Density media */
+               *g = &floppy_type[3];
+       else if (fs->head_number == 2) /* double-sided */
+               *g = &floppy_type[2];
+       else
+               *g = &floppy_type[1];
+
+       return 0;
+}
+
+static void setup_medium(int drive)
+{
+       struct floppy_state *fs;
+       fs = &unit[drive];
+
+       if (swim_readbit(DISK_IN)) {
+               struct floppy_struct *g;
+               fs->disk_in = 1;
+               fs->write_protected = swim_readbit(WRITE_PROT);
+               fs->type = swim_readbit(ONEMEG_MEDIA);
+
+               if (swim_track00())
+                       printk(KERN_ERR
+                               "SWIM: cannot move floppy head to track 0\n");
+
+               swim_track00();
+
+               get_floppy_geometry(drive, 0, &g);
+               fs->total_secs = g->size;
+               fs->secpercyl = g->head * g->sect;
+               fs->secpertrack = g->sect;
+               fs->track = 0;
+       } else {
+               fs->disk_in = 0;
+       }
+}
+
+static int floppy_open(struct block_device *bdev, fmode_t mode)
+{
+       int drive = MINOR(bdev->bd_dev) & 3;
+       struct floppy_state *fs;
+       int err;
+
+       if (drive >= floppy_count)
+               return -ENODEV;
+
+       fs = &unit[drive];
+
+       if (fs->ref_count == -1 || (fs->ref_count && mode & FMODE_EXCL))
+               return -EBUSY;
+
+       if (mode & FMODE_EXCL)
+               fs->ref_count = -1;
+       else
+               fs->ref_count++;
+
+       swim_write(setup, S_IBM_DRIVE  | S_FCLK_DIV2);
+       udelay(10);
+       swim_drive(INTERNAL_DRIVE);
+       swim_motor(ON);
+       swim_action(SETMFM);
+       if (fs->ejected)
+               setup_medium(drive);
+       if (!fs->disk_in) {
+               err = -ENXIO;
+               goto out;
+       }
+
+       if (mode & FMODE_NDELAY)
+               return 0;
+
+       if (mode & (FMODE_READ|FMODE_WRITE)) {
+               check_disk_change(bdev);
+               if ((mode & FMODE_WRITE) && fs->write_protected) {
+                       err = -EROFS;
+                       goto out;
+               }
+       }
+       return 0;
+out:
+       if (fs->ref_count < 0)
+               fs->ref_count = 0;
+       else if (fs->ref_count > 0)
+               --fs->ref_count;
+
+       if (fs->ref_count == 0)
+               swim_motor(OFF);
+       return err;
+}
+
+static int floppy_release(struct gendisk *disk, fmode_t mode)
+{
+       struct floppy_state *fs = &unit[(long)disk->private_data];
+
+       if (fs->ref_count < 0)
+               fs->ref_count = 0;
+       else if (fs->ref_count > 0)
+               --fs->ref_count;
+
+       if (fs->ref_count == 0)
+               swim_motor(OFF);
+
+       return 0;
+}
+
+static int floppy_ioctl(struct block_device *bdev, fmode_t mode,
+                       unsigned int cmd, unsigned long param)
+{
+       struct floppy_state *fs;
+       int err;
+       int drive = (long)bdev->bd_disk->private_data;
+
+       if (drive >= floppy_count)
+               return -ENODEV;
+
+       if ((cmd & 0x80) && !capable(CAP_SYS_ADMIN))
+                       return -EPERM;
+
+       fs = &unit[drive];
+
+       switch (cmd) {
+       case FDEJECT:
+               if (fs->ref_count != 1)
+                       return -EBUSY;
+               err = floppy_eject(fs);
+               return err;
+
+       case FDGETPRM:
+               if (copy_to_user((void *) param, (void *) &floppy_type,
+                                sizeof(struct floppy_struct)))
+                       return -EFAULT;
+               break;
+
+       default:
+               printk(KERN_DEBUG "SWIM floppy_ioctl: unknown cmd %d\n",
+                      cmd);
+               return -ENOSYS;
+       }
+       return 0;
+}
+
+static int floppy_getgeo(struct block_device *bdev, struct hd_geometry *geo)
+{
+       int drive = (long)bdev->bd_disk->private_data;
+       struct floppy_struct *g;
+       int ret;
+
+       ret = get_floppy_geometry(drive, 0, &g);
+       if (ret)
+               return ret;
+
+       geo->heads = g->head;
+       geo->sectors = g->sect;
+       geo->cylinders = g->track;
+
+       return 0;
+}
+
+static int floppy_check_change(struct gendisk *disk)
+{
+       struct floppy_state *fs;
+       int drive = (long)disk->private_data;
+
+       if (drive >= floppy_count)
+               return 0;
+
+       fs = &unit[drive];
+
+       return fs->ejected;
+}
+
+static int floppy_revalidate(struct gendisk *disk)
+{
+       struct floppy_state *fs;
+       int drive = (long)disk->private_data;
+
+       if (drive >= floppy_count)
+               return 0;
+
+       fs = &unit[drive];
+
+       swim_drive(fs->location);
+
+       if (fs->ejected)
+               setup_medium(drive);
+
+       if (!fs->disk_in) {
+               swim_motor(OFF);
+       } else {
+               fs->ejected = 0;
+       }
+
+       return !fs->disk_in;
+}
+
+static struct block_device_operations floppy_fops = {
+       .owner          = THIS_MODULE,
+       .open           = floppy_open,
+       .release        = floppy_release,
+       .locked_ioctl   = floppy_ioctl,
+       .getgeo         = floppy_getgeo,
+       .media_changed  = floppy_check_change,
+       .revalidate_disk= floppy_revalidate,
+};
+
+static struct kobject *floppy_find(dev_t dev, int *part, void *data)
+{
+       int drive = (*part & 3);
+
+       if (drive > floppy_count)
+               return NULL;
+
+       *part = 0;
+       return get_disk(unit[drive].disk);
+}
+
+static int swim_add_floppy(drive_location_t location)
+{
+       struct floppy_state *fs = &unit[floppy_count];
+
+       fs->location = location;
+
+       swim_drive(location);
+
+       swim_motor(OFF);
+
+       if (swim_readbit(SINGLE_SIDED)) {
+               printk(KERN_INFO "SWIM: drive is single sided\n");
+               fs->head_number = 1;
+       } else
+               fs->head_number = 2;
+       fs->ref_count = 0;
+       fs->ejected = 1;
+
+       floppy_count++;
+
+       return 0;
+}
+
+static int __init swim_floppy_init(void)
+{
+       int err;
+       int drive;
+
+       if (SWIMBase == NULL)
+               return 0;
+
+       printk(KERN_INFO "SWIM floppy driver version %s\n", DRIVER_VERSION);
+
+       /* scan floppy drives */
+
+       swim_mode(1);
+
+       swim_drive(INTERNAL_DRIVE);
+       if (swim_readbit(DRIVE_PRESENT)) {
+               printk("SWIM: internal floppy drive detected\n");
+               swim_add_floppy(INTERNAL_DRIVE);
+       }
+       swim_drive(EXTERNAL_DRIVE);
+       if (swim_readbit(DRIVE_PRESENT)) {
+               printk("SWIM: external floppy drive detected\n");
+               swim_add_floppy(EXTERNAL_DRIVE);
+       }
+
+       /* register floppy drives */
+
+       err = register_blkdev(FLOPPY_MAJOR, "fd");
+       if (err) {
+               printk(KERN_ERR "Unable to get major %d for SWIM floppy\n",
+                      FLOPPY_MAJOR);
+               return -EBUSY;
+       }
+
+       for (drive = 0; drive < floppy_count; drive++) {
+               unit[drive].disk = alloc_disk(1);
+               if (unit[drive].disk == NULL) {
+                       err = -ENOMEM;
+                       goto exit_put_disks;
+               }
+       }
+
+       floppy_queue = blk_init_queue(do_fd_request, &swim_lock);
+       if (!floppy_queue) {
+               err = -ENOMEM;
+               goto exit_put_disks;
+       }
+
+       for (drive = 0; drive < floppy_count; drive++) {
+               unit[drive].disk->flags = GENHD_FL_REMOVABLE;
+               unit[drive].disk->major = FLOPPY_MAJOR;
+               unit[drive].disk->first_minor = drive;
+               sprintf(unit[drive].disk->disk_name, "fd%d", drive);
+               unit[drive].disk->fops = &floppy_fops;
+               unit[drive].disk->private_data = (void*)(long)drive;
+               unit[drive].disk->queue = floppy_queue;
+               set_capacity(unit[drive].disk, 2880);
+               add_disk(unit[drive].disk);
+       }
+
+       blk_register_region(MKDEV(FLOPPY_MAJOR, 0), 256, THIS_MODULE,
+                           floppy_find, NULL, NULL);
+
+       return 0;
+
+exit_put_disks:
+       unregister_blkdev(FLOPPY_MAJOR, "fd");
+       while (drive--)
+               put_disk(unit[drive].disk);
+       return err;
+}
+
+static int __init swim_init(void)
+{
+       printk(KERN_INFO "Inserting SWIM floppy driver\n");
+
+       if (base) {
+               printk(KERN_INFO "SWIM: Setting SWIMBase to 0x%p\n", base);
+               SWIMBase = (struct swim *)base;
+       }
+
+       return swim_floppy_init();
+}
+module_init(swim_init)
+
+static void __exit swim_cleanup(void)
+{
+       int drive;
+
+       printk(KERN_INFO "Removing SWIM floppy driver\n");
+
+       blk_unregister_region(MKDEV(FLOPPY_MAJOR, 0), 256);
+
+       for (drive = 0; drive < floppy_count; drive++) {
+               del_gendisk(unit[drive].disk);
+               put_disk(unit[drive].disk);
+       }
+
+       unregister_blkdev(FLOPPY_MAJOR, "fd");
+
+       blk_cleanup_queue(floppy_queue);
+
+       /* eject floppies */
+
+       for (drive = 0; drive < floppy_count; drive++)
+               floppy_eject(&unit[drive]);
+}
+module_exit(swim_cleanup)
+
+MODULE_DESCRIPTION("Driver for SWIM (Sander. Woz Integrated Machine) floppy 
controller");
+MODULE_LICENSE("GPL");
+MODULE_AUTHOR("Laurent Vivier");
+MODULE_ALIAS_BLOCKDEV_MAJOR(FLOPPY_MAJOR);
Index: linux-2.6/drivers/block/swim_asm.S
===================================================================
--- /dev/null   1970-01-01 00:00:00.000000000 +0000
+++ linux-2.6/drivers/block/swim_asm.S  2008-11-01 06:14:39.000000000 +0100
@@ -0,0 +1,295 @@
+/*
+ * low-level functions for the SWIM floppy controller
+ *
+ * needs assembly language because is very timing dependent
+ * this controller exists only on macintosh 680x0 based
+ *
+ * Copyright (C) 2004,2008 Laurent Vivier <[EMAIL PROTECTED]>
+ *
+ * based on Alastair Bridgewater SWIM analysis, 2001
+ * based on netBSD IWM driver (c) 1997, 1998 Hauke Fath.
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version
+ * 2 of the License, or (at your option) any later version.
+ *
+ * 2004-08-21 (lv) - Initial implementation
+ */
+
+       .equ    ph0L,           0x0000
+       .equ    ph0H,           0x0200
+       .equ    ph1L,           0x0400
+       .equ    ph1H,           0x0600
+       .equ    ph2L,           0x0800
+       .equ    ph2H,           0x0a00
+       .equ    ph3L,           0x0c00
+       .equ    ph3H,           0x0e00
+       .equ    mtrOff,         0x1000
+       .equ    mtrOn,          0x1200
+       .equ    intDrive,       0x1400
+       .equ    extDrive,       0x1600
+       .equ    q6L,            0x1800
+       .equ    q6H,            0x1a00
+       .equ    q7L,            0x1c00
+       .equ    q7H,            0x1e00
+
+       .equ    write_data,     0x0000
+       .equ    write_mark,     0x0200
+       .equ    write_CRC,      0x0400
+       .equ    write_parameter,0x0600
+       .equ    write_phase,    0x0800
+       .equ    write_setup,    0x0a00
+       .equ    write_mode0,    0x0c00
+       .equ    write_mode1,    0x0e00
+       .equ    read_data,      0x1000
+       .equ    read_mark,      0x1200
+       .equ    read_error,     0x1400
+       .equ    read_parameter, 0x1600
+       .equ    read_phase,     0x1800
+       .equ    read_setup,     0x1a00
+       .equ    read_status,    0x1c00
+       .equ    read_handshake, 0x1e00
+
+       .equ    o_side, 0
+       .equ    o_track, 1
+       .equ    o_sector, 2
+       .equ    o_size, 3
+       .equ    o_crc0, 4
+       .equ    o_crc1, 5
+
+       .equ    seek_time, 30000
+       .equ    max_retry, 40
+       .equ    sector_size, 512
+
+       .global swim_mode
+swim_mode:
+       link    %a6, #0
+       moveml  %a0/%d2, [EMAIL PROTECTED]
+
+       movel   SWIMBase, %a0
+
+       tstl    %a6@(0x08)
+       beq     iwm_mode
+
+       /* switch to SWIM mode */
+
+       tstb    %a0@(q7L)
+       tstb    %a0@(mtrOff)
+       tstb    %a0@(q6H)
+       moveb   #0x57, %d2
+       moveb   %d2, %a0@(q7H)
+       moveb   #0x17, %a0@(q7H)
+       moveb   %d2, %a0@(q7H)
+       moveb   %d2, %a0@(q7H)
+       moveml  [EMAIL PROTECTED], %a0/%d2
+       unlk    %a6
+       rts
+iwm_mode:
+       /* switch to IWM mode */
+
+       moveb   #0xf8, %a0@(write_mode0)
+       moveml  [EMAIL PROTECTED], %a0/%d2
+       unlk    %a6
+       rts
+
+
+       .global swim_read_sector_header
+swim_read_sector_header:
+       link    %a6, #0
+       moveml  %d1-%d5/%a0-%a4,[EMAIL PROTECTED]
+       movel   %a6@(0x08), %a4
+       bsr     mfm_read_addrmark
+       moveml  [EMAIL PROTECTED], %d1-%d5/%a0-%a4
+       unlk    %a6
+       rts
+
+sector_address_mark:
+       .byte   0xa1, 0xa1, 0xa1, 0xfe
+sector_data_mark:
+       .byte   0xa1, 0xa1, 0xa1, 0xfb
+
+mfm_read_addrmark:
+       movel   SWIMBase, %a3
+       lea     %a3@(read_handshake), %a2
+       lea     %a3@(read_mark), %a3
+       moveq   #-1, %d0
+       movew   #seek_time, %d2
+
+wait_header_init:
+       tstb    %a3@(read_error - read_mark)
+       moveb   #0x18, %a3@(write_mode0 - read_mark)
+       moveb   #0x01, %a3@(write_mode1 - read_mark)
+       moveb   #0x01, %a3@(write_mode0 - read_mark)
+       tstb    %a3@(read_error - read_mark)
+       moveb   #0x08, %a3@(write_mode1 - read_mark)
+
+       lea     sector_address_mark, %a0
+       moveq   #3, %d1
+
+wait_addr_mark_byte:
+
+       tstb    %a2@
+       dbmi    %d2, wait_addr_mark_byte
+       bpl     header_exit
+
+       moveb   %a3@, %d3
+       cmpb    [EMAIL PROTECTED], %d3
+       dbne    %d1, wait_addr_mark_byte
+       bne     wait_header_init
+
+       moveq   #max_retry, %d2
+
+amark0:        tstb    %a2@
+       dbmi    %d2, amark0
+       bpl     signal_nonyb
+
+       moveb   %a3@, %a4@(o_track)
+
+       moveq   #max_retry, %d2
+
+amark1:        tstb    %a2@
+       dbmi    %d2, amark1
+       bpl     signal_nonyb
+
+       moveb   %a3@, %a4@(o_side)
+
+       moveq   #max_retry, %d2
+
+amark2:        tstb    %a2@
+       dbmi    %d2, amark2
+       bpl     signal_nonyb
+
+       moveb   %a3@, %a4@(o_sector)
+
+       moveq   #max_retry, %d2
+
+amark3:        tstb    %a2@
+       dbmi    %d2, amark3
+       bpl     signal_nonyb
+
+       moveb   %a3@, %a4@(o_size)
+
+       moveq   #max_retry, %d2
+
+crc0:  tstb    %a2@
+       dbmi    %d2, crc0
+       bpl     signal_nonyb
+
+       moveb   %a3@, %a4@(o_crc0)
+
+       moveq   #max_retry, %d2
+
+crc1:  tstb    %a2@
+       dbmi    %d2, crc1
+       bpl     signal_nonyb
+
+       moveb   %a3@, %a4@(o_crc1)
+
+       tstb    %a3@(read_error - read_mark)
+
+header_exit:
+       moveq   #0, %d0
+       moveb   #0x18, %a3@(write_mode0 - read_mark)
+       rts
+signal_nonyb:
+       moveq   #-1, %d0
+       moveb   #0x18, %a3@(write_mode0 - read_mark)
+       rts
+
+       .global swim_read_sector_data
+swim_read_sector_data:
+       link    %a6, #0
+       moveml  %d1-%d5/%a0-%a5,[EMAIL PROTECTED]
+       movel   %a6@(0x08), %a4
+       bsr     mfm_read_data
+       moveml  [EMAIL PROTECTED], %d1-%d5/%a0-%a5
+       unlk    %a6
+       rts
+
+mfm_read_data:
+       movel   SWIMBase, %a3
+       lea     %a3@(read_handshake), %a2
+       lea     %a3@(read_data), %a5
+       lea     %a3@(read_mark), %a3
+       movew   #seek_time, %d2
+
+wait_data_init:
+       tstb    %a3@(read_error - read_mark)
+       moveb   #0x18, %a3@(write_mode0 - read_mark)
+       moveb   #0x01, %a3@(write_mode1 - read_mark)
+       moveb   #0x01, %a3@(write_mode0 - read_mark)
+       tstb    %a3@(read_error - read_mark)
+       moveb   #0x08, %a3@(write_mode1 - read_mark)
+
+       lea     sector_data_mark, %a0
+       moveq   #3, %d1
+
+       /* wait data address mark */
+
+wait_data_mark_byte:
+
+       tstb    %a2@
+       dbmi    %d2, wait_data_mark_byte
+       bpl     data_exit
+
+       moveb   %a3@, %d3
+       cmpb    [EMAIL PROTECTED], %d3
+       dbne    %d1, wait_data_mark_byte
+       bne     wait_data_init
+
+       /* read data */
+
+       tstb    %a3@(read_error - read_mark)
+
+       movel   #sector_size-1, %d4             /* sector size */
+read_new_data:
+       movew   #max_retry, %d2
+read_data_loop:
+       moveb   %a2@, %d5
+       andb    #0xc0, %d5
+       dbne    %d2, read_data_loop
+       beq     data_exit
+       moveb   %a5@, [EMAIL PROTECTED]
+       andb    #0x40, %d5
+       dbne    %d4, read_new_data
+       beq     exit_loop
+       moveb   %a5@, [EMAIL PROTECTED]
+       dbra    %d4, read_new_data
+exit_loop:
+
+       /* read CRC */
+
+       movew   #max_retry, %d2
+data_crc0:
+
+       tstb    %a2@
+       dbmi    %d2, data_crc0
+       bpl     data_exit
+
+       moveb   %a3@, %d5
+
+       moveq   #max_retry, %d2
+
+data_crc1:
+
+       tstb    %a2@
+       dbmi    %d2, data_crc1
+       bpl     data_exit
+
+       moveb   %a3@, %d5
+
+       tstb    %a3@(read_error - read_mark)
+
+       moveb   #0x18, %a3@(write_mode0 - read_mark)
+
+       /* return number of bytes read */
+
+       movel   #sector_size, %d0
+       addw    #1, %d4
+       subl    %d4, %d0
+       rts
+data_exit:
+       moveb   #0x18, %a3@(write_mode0 - read_mark)
+       moveq   #-1, %d0
+       rts
--
To unsubscribe from this list: send the line "unsubscribe linux-m68k" in
the body of a message to [EMAIL PROTECTED]
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to