here is the updated version of the floppy controller to I2C
adapter driver, which uses the Motor Control and Disk Changed
lines to interface to an I2C bus.

I hope I fixed the style issues (except for the access to 
global var from "remove" function - to keep it similar to
the i2c-parport-light for now)

of course the patch is compile and run tested on PC style
hardware with my test circuit connecting to an LM75 and
LM85 (temp/fan/pwm)

please consider for (mainline) inclusion!

TIA,
Herbert


Signed-off-by: Herbert Poetzl <[EMAIL PROTECTED]>

diff -NurpP --minimal linux-2.6.27-rc3/Documentation/i2c/busses/i2c-floppy 
linux-2.6.27-rc3-fi2c-v0.2/Documentation/i2c/busses/i2c-floppy
--- linux-2.6.27-rc3/Documentation/i2c/busses/i2c-floppy        1970-01-01 
01:00:00.000000000 +0100
+++ linux-2.6.27-rc3-fi2c-v0.2/Documentation/i2c/busses/i2c-floppy      
2008-08-18 04:34:53.000000000 +0200
@@ -0,0 +1,38 @@
+Kernel driver i2c-floppy
+
+Author: Herbert Poetzl <[EMAIL PROTECTED]>
+
+This driver is for a simple do-it-yourself floppy controller to
+I2C adapters which uses direct I/O access to control both Motor
+Enable lines (A/B) and read back data via the Disk Changed line.
+
+The following circuit is not suited to be used together with
+a Floppy drive (which would need some logic to work properly)
+but as replacement.
+
+ +---+
+ | F |  Motor Enable A [/MOTEA,10] -------------- SCL
+ | L |
+ | O |  Motor Enable B [/MOTEB,16] ------+------- SDA
+ | P |                                   |
+ | P |                                   |
+ | Y |  Disk Change   [/DSKCHG,34] ------+    +-- GND
+ +-+-+                                        |
+   |                                          |
+  GND                                       GND
+
+the Motor Enable output of Floppy Disk Controllers is an open
+drain output (48mA), usually pulled up to +5V via 1k or higher,
+and the Disk Change is a Schmitt Trigger (0.8V/2.2V) input with
+max 150uA input current (data taken from the WD/FDC 37C6xx
+Floppy Disk Subsystem Controller datasheets, which basically
+combine all the necessary parts of the PC Floppy Circuit)
+
+so it should be fine to connect it to all 5V i2c devices, and
+most 3.3V devices (which are usually capable of handling 5V
+i2c bus voltages)
+
+power for those devices can be drawn from either the Floppy
+power connector (+5V/+12V) or from the SATA 3.3V lines
+(or if you prefer from some external source)
+
diff -NurpP --minimal linux-2.6.27-rc3/drivers/i2c/busses/i2c-floppy.c 
linux-2.6.27-rc3-fi2c-v0.2/drivers/i2c/busses/i2c-floppy.c
--- linux-2.6.27-rc3/drivers/i2c/busses/i2c-floppy.c    1970-01-01 
01:00:00.000000000 +0100
+++ linux-2.6.27-rc3-fi2c-v0.2/drivers/i2c/busses/i2c-floppy.c  2008-08-18 
03:35:34.000000000 +0200
@@ -0,0 +1,261 @@
+/* ------------------------------------------------------------------------ *
+ * i2c-floppy.c I2C bus over floppy controller                              *
+ * ------------------------------------------------------------------------ *
+   Copyright (C) 2008 Herbert Poetzl <[EMAIL PROTECTED]>
+
+   Somewhat based on i2c-parport-light.c driver
+   Copyright (C) 2003-2007 Jean Delvare <[EMAIL PROTECTED]>
+
+   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.
+
+   This program is distributed in the hope that it will be useful,
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+   GNU General Public License for more details.
+
+   You should have received a copy of the GNU General Public License
+   along with this program; if not, write to the Free Software
+   Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
+ * ------------------------------------------------------------------------ */
+
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/init.h>
+#include <linux/platform_device.h>
+#include <linux/ioport.h>
+#include <linux/i2c.h>
+#include <linux/i2c-algo-bit.h>
+#include <linux/io.h>
+
+static struct platform_device *pdev;
+static unsigned char dor;
+
+static u16 base;
+module_param(base, ushort, 0);
+MODULE_PARM_DESC(base, "Base I/O address");
+
+
+#define DEFAULT_BASE   0x3F0           /* for PC style hardware */
+#define DRVNAME                "i2c-floppy"
+
+
+#define FOFF_DOR       0x02
+#define FOFF_DIR       0x07
+
+#define FDOR_MOTEA     0x10
+#define FDOR_MOTEB     0x20
+
+#define FDIR_DCHNG     0x80
+
+#define SCL            FDOR_MOTEA
+#define SDA            FDOR_MOTEB
+#define SDA_IN         FDIR_DCHNG
+
+#define LO_INV         (SDA|SCL)
+#define LI_INV         (SDA_IN)
+
+
+/* ----- Device list ------------------------------------------------------ */
+
+#if 0
+struct i2c_floppy_data {
+       unsigned char dor;
+};
+
+struct i2c_floppy {
+       struct i2c_adapter adapter;
+       struct i2c_algo_bit_data algo_data;
+       struct i2c_floppy_data data;
+       struct i2c_floppy *next;
+};
+#endif
+
+
+/* ----- Low-level floppy access ------------------------------------------ */
+
+static inline void port_dor_out(unsigned char d)
+{
+       outb(d ^ LO_INV, base + FOFF_DOR);
+}
+
+static inline unsigned char port_dir_in(void)
+{
+       return inb(base + FOFF_DIR) ^ LI_INV;
+}
+
+
+/* ----- I2C algorithm call-back functions and structures ----------------- */
+
+static void floppy_setscl(void *data, int state)
+{
+       if (state)
+               dor |= SCL;
+       else
+               dor &= ~SCL;
+
+       port_dor_out(dor);
+}
+
+static void floppy_setsda(void *data, int state)
+{
+       if (state)
+               dor |= SDA;
+       else
+               dor &= ~SDA;
+
+       port_dor_out(dor);
+}
+
+static int floppy_getsda(void *data)
+{
+       return port_dir_in() & SDA_IN;
+}
+
+/* Encapsulate the functions above in the correct structure
+   Note that getscl is set to NULL because SCL cannot be read
+   back with the current driver */
+static struct i2c_algo_bit_data floppy_algo_data = {
+       .setsda         = floppy_setsda,
+       .setscl         = floppy_setscl,
+       .getsda         = floppy_getsda,
+       .udelay         = 50,
+       .timeout        = HZ,
+};
+
+
+/* ----- Driver registration ---------------------------------------------- */
+
+static struct i2c_adapter floppy_adapter = {
+       .owner          = THIS_MODULE,
+       .class          = I2C_CLASS_HWMON,
+       .algo_data      = &floppy_algo_data,
+       .name           = "Floppy controller adapter",
+};
+
+static int __devinit i2c_floppy_probe(struct platform_device *pdev)
+{
+       int err;
+       struct resource *res;
+
+       res = platform_get_resource(pdev, IORESOURCE_IO, 0);
+       if (!request_region(res->start, res->end - res->start + 1, DRVNAME))
+               return -EBUSY;
+
+       /* Reset hardware to a sane state (SCL and SDA high) */
+       floppy_setsda(NULL, 1);
+       floppy_setscl(NULL, 1);
+
+       floppy_adapter.dev.parent = &pdev->dev;
+       err = i2c_bit_add_bus(&floppy_adapter);
+       if (err) {
+               dev_err(&pdev->dev, "Unable to register with I2C\n");
+               goto exit_region;
+       }
+       return 0;
+
+exit_region:
+       release_region(res->start, res->end - res->start + 1);
+       return err;
+}
+
+static int __devexit i2c_floppy_remove(struct platform_device *pdev)
+{
+       struct resource *res;
+
+       i2c_del_adapter(&floppy_adapter);
+
+       res = platform_get_resource(pdev, IORESOURCE_IO, 0);
+       release_region(res->start, res->end - res->start + 1);
+       return 0;
+}
+
+static struct platform_driver i2c_floppy_driver = {
+       .driver = {
+               .owner  = THIS_MODULE,
+               .name   = DRVNAME,
+       },
+       .probe          = i2c_floppy_probe,
+       .remove         = __devexit_p(i2c_floppy_remove),
+};
+
+static int __init i2c_floppy_device_add(u16 address)
+{
+       struct resource res = {
+               .start  = address,
+               .end    = address + 7,
+               .name   = DRVNAME,
+               .flags  = IORESOURCE_IO,
+       };
+       int err;
+
+       pdev = platform_device_alloc(DRVNAME, -1);
+       if (!pdev) {
+               err = -ENOMEM;
+               printk(KERN_ERR DRVNAME ": Device allocation failed\n");
+               goto exit;
+       }
+
+       err = platform_device_add_resources(pdev, &res, 1);
+       if (err) {
+               printk(KERN_ERR DRVNAME ": Device resource addition failed "
+                      "(%d)\n", err);
+               goto exit_device_put;
+       }
+
+       err = platform_device_add(pdev);
+       if (err) {
+               printk(KERN_ERR DRVNAME ": Device addition failed (%d)\n",
+                      err);
+               goto exit_device_put;
+       }
+
+       return 0;
+
+exit_device_put:
+       platform_device_put(pdev);
+exit:
+       return err;
+}
+
+static int __init i2c_floppy_init(void)
+{
+       int err;
+
+       if (base == 0) {
+               pr_info(DRVNAME ": using default base 0x%x\n", DEFAULT_BASE);
+               base = DEFAULT_BASE;
+       }
+
+       /* Sets global pdev as a side effect */
+       err = i2c_floppy_device_add(base);
+       if (err)
+               goto exit;
+
+       err = platform_driver_register(&i2c_floppy_driver);
+       if (err)
+               goto exit_device;
+
+       return 0;
+
+exit_device:
+       platform_device_unregister(pdev);
+exit:
+       return err;
+}
+
+static void __exit i2c_floppy_exit(void)
+{
+       platform_driver_unregister(&i2c_floppy_driver);
+       platform_device_unregister(pdev);
+}
+
+
+MODULE_AUTHOR("Herbert Poetzl <[EMAIL PROTECTED]>");
+MODULE_DESCRIPTION("I2C bus over floppy controller");
+MODULE_LICENSE("GPL");
+
+module_init(i2c_floppy_init);
+module_exit(i2c_floppy_exit);
diff -NurpP --minimal linux-2.6.27-rc3/drivers/i2c/busses/Kconfig 
linux-2.6.27-rc3-fi2c-v0.2/drivers/i2c/busses/Kconfig
--- linux-2.6.27-rc3/drivers/i2c/busses/Kconfig 2008-08-15 21:19:24.000000000 
+0200
+++ linux-2.6.27-rc3-fi2c-v0.2/drivers/i2c/busses/Kconfig       2008-08-18 
03:46:31.000000000 +0200
@@ -490,6 +490,22 @@ config I2C_VERSATILE
 
 comment "External I2C/SMBus adapter drivers"
 
+config I2C_FLOPPY
+       tristate "Floppy controller adapter"
+       select I2C_ALGOBIT
+       default n
+       help
+         This supports a simple do-it-yourself floppy controller to
+         I2C adapters using the motor control lines for SDA and SCL,
+         and the drive change input for SDA readback.
+
+         This support is also available as a module.  If so, the module
+         will be called i2c-floppy.
+
+         If you do not have such a device, and do not plan to build one,
+         it's safe to say N here. Do not say Y here and to the floppy
+         driver unless you know exactly what you are doing.
+
 config I2C_PARPORT
        tristate "Parallel port adapter"
        depends on PARPORT
diff -NurpP --minimal linux-2.6.27-rc3/drivers/i2c/busses/Makefile 
linux-2.6.27-rc3-fi2c-v0.2/drivers/i2c/busses/Makefile
--- linux-2.6.27-rc3/drivers/i2c/busses/Makefile        2008-08-15 
21:19:24.000000000 +0200
+++ linux-2.6.27-rc3-fi2c-v0.2/drivers/i2c/busses/Makefile      2008-08-18 
03:46:42.000000000 +0200
@@ -48,6 +48,7 @@ obj-$(CONFIG_I2C_SIMTEC)      += i2c-simtec.o
 obj-$(CONFIG_I2C_VERSATILE)    += i2c-versatile.o
 
 # External I2C/SMBus adapter drivers
+obj-$(CONFIG_I2C_FLOPPY)       += i2c-floppy.o
 obj-$(CONFIG_I2C_PARPORT)      += i2c-parport.o
 obj-$(CONFIG_I2C_PARPORT_LIGHT)        += i2c-parport-light.o
 obj-$(CONFIG_I2C_TAOS_EVM)     += i2c-taos-evm.o

_______________________________________________
i2c mailing list
[email protected]
http://lists.lm-sensors.org/mailman/listinfo/i2c

Reply via email to