From: Laurent Vivier <Laurent@xxxxxxxxxxxx> 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 <Laurent@xxxxxxxxxxxx> --- 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 <Laurent@xxxxxxxxxxxx> + * + * 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 <Laurent@xxxxxxxxxxxx> + * + * 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 <Laurent@xxxxxxxxxxxx> + * + * 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, %sp@- + + 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 %sp@+, %a0/%d2 + unlk %a6 + rts +iwm_mode: + /* switch to IWM mode */ + + moveb #0xf8, %a0@(write_mode0) + moveml %sp@+, %a0/%d2 + unlk %a6 + rts + + + .global swim_read_sector_header +swim_read_sector_header: + link %a6, #0 + moveml %d1-%d5/%a0-%a4,%sp@- + movel %a6@(0x08), %a4 + bsr mfm_read_addrmark + moveml %sp@+, %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 %a0@+, %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,%sp@- + movel %a6@(0x08), %a4 + bsr mfm_read_data + moveml %sp@+, %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 %a0@+, %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@, %a4@+ + andb #0x40, %d5 + dbne %d4, read_new_data + beq exit_loop + moveb %a5@, %a4@+ + 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 majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html