2006-01-11 22:23:49 +03:00
/*
2011-06-06 11:16:30 +04:00
* parport - to - butterfly adapter
2006-01-11 22:23:49 +03:00
*
* Copyright ( C ) 2005 David Brownell
*
* 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 0213 9 , USA .
*/
# include <linux/kernel.h>
# include <linux/init.h>
# include <linux/delay.h>
2011-07-03 23:44:29 +04:00
# include <linux/module.h>
2007-05-08 11:27:42 +04:00
# include <linux/device.h>
2006-01-11 22:23:49 +03:00
# include <linux/parport.h>
2006-10-18 21:55:46 +04:00
# include <linux/sched.h>
2006-01-11 22:23:49 +03:00
# include <linux/spi/spi.h>
# include <linux/spi/spi_bitbang.h>
# include <linux/spi/flash.h>
# include <linux/mtd/partitions.h>
/*
* This uses SPI to talk with an " AVR Butterfly " , which is a $ US20 card
* with a battery powered AVR microcontroller and lots of goodies . You
* can use GCC to develop firmware for this .
*
* See Documentation / spi / butterfly for information about how to build
* and use this custom parallel port cable .
*/
/* DATA output bits (pins 2..9 == D0..D7) */
# define butterfly_nreset (1 << 1) /* pin 3 */
# define spi_sck_bit (1 << 0) /* pin 2 */
# define spi_mosi_bit (1 << 7) /* pin 9 */
# define vcc_bits ((1 << 6) | (1 << 5)) /* pins 7, 8 */
/* STATUS input bits */
# define spi_miso_bit PARPORT_STATUS_BUSY /* pin 11 */
/* CONTROL output bits */
# define spi_cs_bit PARPORT_CONTROL_SELECT /* pin 17 */
static inline struct butterfly * spidev_to_pp ( struct spi_device * spi )
{
return spi - > controller_data ;
}
struct butterfly {
/* REVISIT ... for now, this must be first */
struct spi_bitbang bitbang ;
struct parport * port ;
struct pardevice * pd ;
u8 lastbyte ;
struct spi_device * dataflash ;
struct spi_device * butterfly ;
struct spi_board_info info [ 2 ] ;
} ;
/*----------------------------------------------------------------------*/
static inline void
setsck ( struct spi_device * spi , int is_on )
{
struct butterfly * pp = spidev_to_pp ( spi ) ;
u8 bit , byte = pp - > lastbyte ;
2007-05-08 11:32:13 +04:00
bit = spi_sck_bit ;
2006-01-11 22:23:49 +03:00
if ( is_on )
byte | = bit ;
else
byte & = ~ bit ;
parport_write_data ( pp - > port , byte ) ;
pp - > lastbyte = byte ;
}
static inline void
setmosi ( struct spi_device * spi , int is_on )
{
struct butterfly * pp = spidev_to_pp ( spi ) ;
u8 bit , byte = pp - > lastbyte ;
2007-05-08 11:32:13 +04:00
bit = spi_mosi_bit ;
2006-01-11 22:23:49 +03:00
if ( is_on )
byte | = bit ;
else
byte & = ~ bit ;
parport_write_data ( pp - > port , byte ) ;
pp - > lastbyte = byte ;
}
static inline int getmiso ( struct spi_device * spi )
{
struct butterfly * pp = spidev_to_pp ( spi ) ;
int value ;
u8 bit ;
2007-05-08 11:32:13 +04:00
bit = spi_miso_bit ;
2006-01-11 22:23:49 +03:00
/* only STATUS_BUSY is NOT negated */
value = ! ( parport_read_status ( pp - > port ) & bit ) ;
return ( bit = = PARPORT_STATUS_BUSY ) ? value : ! value ;
}
static void butterfly_chipselect ( struct spi_device * spi , int value )
{
struct butterfly * pp = spidev_to_pp ( spi ) ;
/* set default clock polarity */
2006-01-22 00:21:43 +03:00
if ( value ! = BITBANG_CS_INACTIVE )
2006-01-11 22:23:49 +03:00
setsck ( spi , spi - > mode & SPI_CPOL ) ;
2006-01-22 00:21:43 +03:00
/* here, value == "activate or not";
* most PARPORT_CONTROL_ * bits are negated , so we must
* morph it to value = = " bit value to write in control register "
*/
2006-01-11 22:23:49 +03:00
if ( spi_cs_bit = = PARPORT_CONTROL_INIT )
value = ! value ;
parport_frob_control ( pp - > port , spi_cs_bit , value ? spi_cs_bit : 0 ) ;
}
/* we only needed to implement one mode here, and choose SPI_MODE_0 */
2013-10-14 05:34:24 +04:00
# define spidelay(X) do { } while (0)
/* #define spidelay ndelay */
2006-01-11 22:23:49 +03:00
2011-06-06 11:16:30 +04:00
# include "spi-bitbang-txrx.h"
2006-01-11 22:23:49 +03:00
static u32
butterfly_txrx_word_mode0 ( struct spi_device * spi ,
unsigned nsecs ,
u32 word , u8 bits )
{
2010-07-01 00:27:32 +04:00
return bitbang_txrx_be_cpha0 ( spi , nsecs , 0 , 0 , word , bits ) ;
2006-01-11 22:23:49 +03:00
}
/*----------------------------------------------------------------------*/
/* override default partitioning with cmdlinepart */
static struct mtd_partition partitions [ ] = { {
2006-01-22 00:21:43 +03:00
/* JFFS2 wants partitions of 4*N blocks for this device,
* so sectors 0 and 1 can ' t be partitions by themselves .
*/
2006-01-11 22:23:49 +03:00
/* sector 0 = 8 pages * 264 bytes/page (1 block)
* sector 1 = 248 pages * 264 bytes / page
*/
2013-10-14 05:34:24 +04:00
. name = " bookkeeping " , /* 66 KB */
2006-01-11 22:23:49 +03:00
. offset = 0 ,
. size = ( 8 + 248 ) * 264 ,
2013-10-14 05:34:24 +04:00
/* .mask_flags = MTD_WRITEABLE, */
2006-01-11 22:23:49 +03:00
} , {
/* sector 2 = 256 pages * 264 bytes/page
* sectors 3 - 5 = 512 pages * 264 bytes / page
*/
2013-10-14 05:34:24 +04:00
. name = " filesystem " , /* 462 KB */
2006-01-11 22:23:49 +03:00
. offset = MTDPART_OFS_APPEND ,
. size = MTDPART_SIZ_FULL ,
} } ;
static struct flash_platform_data flash = {
. name = " butterflash " ,
. parts = partitions ,
. nr_parts = ARRAY_SIZE ( partitions ) ,
} ;
/* REVISIT remove this ugly global and its "only one" limitation */
static struct butterfly * butterfly ;
static void butterfly_attach ( struct parport * p )
{
struct pardevice * pd ;
int status ;
struct butterfly * pp ;
struct spi_master * master ;
2007-05-08 11:27:42 +04:00
struct device * dev = p - > physport - > dev ;
2006-01-11 22:23:49 +03:00
2007-05-08 11:27:42 +04:00
if ( butterfly | | ! dev )
2006-01-11 22:23:49 +03:00
return ;
/* REVISIT: this just _assumes_ a butterfly is there ... no probe,
* and no way to be selective about what it binds to .
*/
2013-10-14 05:34:24 +04:00
master = spi_alloc_master ( dev , sizeof ( * pp ) ) ;
2006-01-11 22:23:49 +03:00
if ( ! master ) {
status = - ENOMEM ;
goto done ;
}
pp = spi_master_get_devdata ( master ) ;
/*
* SPI and bitbang hookup
*
* use default setup ( ) , cleanup ( ) , and transfer ( ) methods ; and
* only bother implementing mode 0. Start it later .
*/
master - > bus_num = 42 ;
master - > num_chipselect = 2 ;
2013-09-10 11:43:41 +04:00
pp - > bitbang . master = master ;
2006-01-11 22:23:49 +03:00
pp - > bitbang . chipselect = butterfly_chipselect ;
pp - > bitbang . txrx_word [ SPI_MODE_0 ] = butterfly_txrx_word_mode0 ;
/*
* parport hookup
*/
pp - > port = p ;
pd = parport_register_device ( p , " spi_butterfly " ,
NULL , NULL , NULL ,
0 /* FLAGS */ , pp ) ;
if ( ! pd ) {
status = - ENOMEM ;
goto clean0 ;
}
pp - > pd = pd ;
status = parport_claim ( pd ) ;
if ( status < 0 )
goto clean1 ;
/*
* Butterfly reset , powerup , run firmware
*/
pr_debug ( " %s: powerup/reset Butterfly \n " , p - > name ) ;
/* nCS for dataflash (this bit is inverted on output) */
parport_frob_control ( pp - > port , spi_cs_bit , 0 ) ;
/* stabilize power with chip in reset (nRESET), and
2007-05-08 11:32:13 +04:00
* spi_sck_bit clear ( CPOL = 0 )
2006-01-11 22:23:49 +03:00
*/
pp - > lastbyte | = vcc_bits ;
parport_write_data ( pp - > port , pp - > lastbyte ) ;
msleep ( 5 ) ;
/* take it out of reset; assume long reset delay */
pp - > lastbyte | = butterfly_nreset ;
parport_write_data ( pp - > port , pp - > lastbyte ) ;
msleep ( 100 ) ;
/*
* Start SPI . . . for now , hide that we ' re two physical busses .
*/
status = spi_bitbang_start ( & pp - > bitbang ) ;
if ( status < 0 )
goto clean2 ;
2006-01-22 00:21:43 +03:00
/* Bus 1 lets us talk to at45db041b (firmware disables AVR SPI), AVR
* ( firmware resets at45 , acts as spi slave ) or neither ( we ignore
* both , AVR uses AT45 ) . Here we expect firmware for the first option .
2006-01-11 22:23:49 +03:00
*/
2006-05-21 02:00:17 +04:00
2006-01-11 22:23:49 +03:00
pp - > info [ 0 ] . max_speed_hz = 15 * 1000 * 1000 ;
strcpy ( pp - > info [ 0 ] . modalias , " mtd_dataflash " ) ;
pp - > info [ 0 ] . platform_data = & flash ;
pp - > info [ 0 ] . chip_select = 1 ;
pp - > info [ 0 ] . controller_data = pp ;
pp - > dataflash = spi_new_device ( pp - > bitbang . master , & pp - > info [ 0 ] ) ;
if ( pp - > dataflash )
pr_debug ( " %s: dataflash at %s \n " , p - > name ,
2009-01-06 21:44:37 +03:00
dev_name ( & pp - > dataflash - > dev ) ) ;
2006-01-11 22:23:49 +03:00
pr_info ( " %s: AVR Butterfly \n " , p - > name ) ;
butterfly = pp ;
return ;
clean2 :
/* turn off VCC */
parport_write_data ( pp - > port , 0 ) ;
parport_release ( pp - > pd ) ;
clean1 :
parport_unregister_device ( pd ) ;
clean0 :
( void ) spi_master_put ( pp - > bitbang . master ) ;
done :
pr_debug ( " %s: butterfly probe, fail %d \n " , p - > name , status ) ;
}
static void butterfly_detach ( struct parport * p )
{
struct butterfly * pp ;
int status ;
/* FIXME this global is ugly ... but, how to quickly get from
* the parport to the " struct butterfly " associated with it ?
* " old school " driver - internal device lists ?
*/
if ( ! butterfly | | butterfly - > port ! = p )
return ;
pp = butterfly ;
butterfly = NULL ;
2006-01-22 00:21:43 +03:00
/* stop() unregisters child devices too */
2006-01-11 22:23:49 +03:00
status = spi_bitbang_stop ( & pp - > bitbang ) ;
/* turn off VCC */
parport_write_data ( pp - > port , 0 ) ;
msleep ( 10 ) ;
parport_release ( pp - > pd ) ;
parport_unregister_device ( pp - > pd ) ;
( void ) spi_master_put ( pp - > bitbang . master ) ;
}
static struct parport_driver butterfly_driver = {
. name = " spi_butterfly " ,
. attach = butterfly_attach ,
. detach = butterfly_detach ,
} ;
static int __init butterfly_init ( void )
{
return parport_register_driver ( & butterfly_driver ) ;
}
device_initcall ( butterfly_init ) ;
static void __exit butterfly_exit ( void )
{
parport_unregister_driver ( & butterfly_driver ) ;
}
module_exit ( butterfly_exit ) ;
2006-01-22 00:21:43 +03:00
MODULE_DESCRIPTION ( " Parport Adapter driver for AVR Butterfly " ) ;
2006-01-11 22:23:49 +03:00
MODULE_LICENSE ( " GPL " ) ;