2019-06-04 11:11:33 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2023-04-23 08:48:45 +03:00
/*
2007-07-20 21:22:57 +04:00
* arch / arm / mac - sa1100 / jornada720_ssp . c
*
* Copyright ( C ) 2006 / 2007 Kristoffer Ericson < Kristoffer . Ericson @ gmail . com >
* Copyright ( C ) 2006 Filip Zyzniewski < filip . zyzniewski @ tefnet . pl >
*
* SSP driver for the HP Jornada 710 / 720 / 728
*/
# include <linux/delay.h>
# include <linux/errno.h>
# include <linux/init.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/platform_device.h>
# include <linux/sched.h>
2012-06-06 14:42:36 +04:00
# include <linux/io.h>
2007-07-20 21:22:57 +04:00
2008-08-05 19:14:15 +04:00
# include <mach/hardware.h>
# include <mach/jornada720.h>
2008-09-18 15:19:39 +04:00
# include <asm/hardware/ssp.h>
2007-07-20 21:22:57 +04:00
static DEFINE_SPINLOCK ( jornada_ssp_lock ) ;
static unsigned long jornada_ssp_flags ;
/**
* jornada_ssp_reverse - reverses input byte
2023-04-23 08:48:45 +03:00
* @ byte : input byte to reverse
2007-07-20 21:22:57 +04:00
*
2011-03-31 05:57:33 +04:00
* we need to reverse all data we receive from the mcu due to its physical location
2007-07-20 21:22:57 +04:00
* returns : 01110111 - > 11101110
*/
2017-07-13 00:37:08 +03:00
inline u8 jornada_ssp_reverse ( u8 byte )
2007-07-20 21:22:57 +04:00
{
return
( ( 0x80 & byte ) > > 7 ) |
( ( 0x40 & byte ) > > 5 ) |
( ( 0x20 & byte ) > > 3 ) |
( ( 0x10 & byte ) > > 1 ) |
( ( 0x08 & byte ) < < 1 ) |
( ( 0x04 & byte ) < < 3 ) |
( ( 0x02 & byte ) < < 5 ) |
( ( 0x01 & byte ) < < 7 ) ;
} ;
EXPORT_SYMBOL ( jornada_ssp_reverse ) ;
/**
* jornada_ssp_byte - waits for ready ssp bus and sends byte
2023-04-23 08:48:45 +03:00
* @ byte : input byte to transmit
2007-07-20 21:22:57 +04:00
*
* waits for fifo buffer to clear and then transmits , if it doesn ' t then we will
* timeout after < timeout > rounds . Needs mcu running before its called .
*
* returns : % mcu output on success
2009-06-04 18:20:28 +04:00
* : % - ETIMEDOUT on timeout
2007-07-20 21:22:57 +04:00
*/
int jornada_ssp_byte ( u8 byte )
{
int timeout = 400000 ;
u16 ret ;
while ( ( GPLR & GPIO_GPIO10 ) ) {
if ( ! - - timeout ) {
printk ( KERN_WARNING " SSP: timeout while waiting for transmit \n " ) ;
return - ETIMEDOUT ;
}
cpu_relax ( ) ;
}
ret = jornada_ssp_reverse ( byte ) < < 8 ;
ssp_write_word ( ret ) ;
ssp_read_word ( & ret ) ;
return jornada_ssp_reverse ( ret ) ;
} ;
EXPORT_SYMBOL ( jornada_ssp_byte ) ;
/**
* jornada_ssp_inout - decide if input is command or trading byte
2023-04-23 08:48:45 +03:00
* @ byte : input byte to send ( may be % TXDUMMY )
2007-07-20 21:22:57 +04:00
*
* returns : ( jornada_ssp_byte ( byte ) ) on success
2009-06-04 18:20:28 +04:00
* : % - ETIMEDOUT on timeout failure
2007-07-20 21:22:57 +04:00
*/
int jornada_ssp_inout ( u8 byte )
{
int ret , i ;
/* true means command byte */
if ( byte ! = TXDUMMY ) {
ret = jornada_ssp_byte ( byte ) ;
/* Proper return to commands is TxDummy */
if ( ret ! = TXDUMMY ) {
for ( i = 0 ; i < 256 ; i + + ) /* flushing bus */
if ( jornada_ssp_byte ( TXDUMMY ) = = - 1 )
break ;
return - ETIMEDOUT ;
}
} else /* Exchange TxDummy for data */
ret = jornada_ssp_byte ( TXDUMMY ) ;
return ret ;
} ;
EXPORT_SYMBOL ( jornada_ssp_inout ) ;
/**
* jornada_ssp_start - enable mcu
*
*/
2008-09-18 15:19:39 +04:00
void jornada_ssp_start ( void )
2007-07-20 21:22:57 +04:00
{
spin_lock_irqsave ( & jornada_ssp_lock , jornada_ssp_flags ) ;
GPCR = GPIO_GPIO25 ;
udelay ( 50 ) ;
2008-09-18 15:19:39 +04:00
return ;
2007-07-20 21:22:57 +04:00
} ;
EXPORT_SYMBOL ( jornada_ssp_start ) ;
/**
* jornada_ssp_end - disable mcu and turn off lock
*
*/
2008-09-18 15:19:39 +04:00
void jornada_ssp_end ( void )
2007-07-20 21:22:57 +04:00
{
GPSR = GPIO_GPIO25 ;
spin_unlock_irqrestore ( & jornada_ssp_lock , jornada_ssp_flags ) ;
2008-09-18 15:19:39 +04:00
return ;
2007-07-20 21:22:57 +04:00
} ;
EXPORT_SYMBOL ( jornada_ssp_end ) ;
2012-12-22 02:02:24 +04:00
static int jornada_ssp_probe ( struct platform_device * dev )
2007-07-20 21:22:57 +04:00
{
int ret ;
GPSR = GPIO_GPIO25 ;
ret = ssp_init ( ) ;
/* worked fine, lets not bother with anything else */
if ( ! ret ) {
printk ( KERN_INFO " SSP: device initialized with irq \n " ) ;
return ret ;
}
printk ( KERN_WARNING " SSP: initialization failed, trying non-irq solution \n " ) ;
/* init of Serial 4 port */
Ser4MCCR0 = 0 ;
Ser4SSCR0 = 0x0387 ;
Ser4SSCR1 = 0x18 ;
/* clear out any left over data */
ssp_flush ( ) ;
/* enable MCU */
jornada_ssp_start ( ) ;
/* see if return value makes sense */
ret = jornada_ssp_inout ( GETBRIGHTNESS ) ;
/* seems like it worked, just feed it with TxDummy to get rid of data */
2007-07-26 10:41:12 +04:00
if ( ret = = TXDUMMY )
2007-07-20 21:22:57 +04:00
jornada_ssp_inout ( TXDUMMY ) ;
jornada_ssp_end ( ) ;
/* failed, lets just kill everything */
if ( ret = = - ETIMEDOUT ) {
printk ( KERN_WARNING " SSP: attempts failed, bailing \n " ) ;
ssp_exit ( ) ;
return - ENODEV ;
}
/* all fine */
printk ( KERN_INFO " SSP: device initialized \n " ) ;
return 0 ;
} ;
2023-03-14 13:32:25 +03:00
static void jornada_ssp_remove ( struct platform_device * dev )
2007-07-20 21:22:57 +04:00
{
2011-03-31 05:57:33 +04:00
/* Note that this doesn't actually remove the driver, since theres nothing to remove
2007-07-20 21:22:57 +04:00
* It just makes sure everything is turned off */
GPSR = GPIO_GPIO25 ;
ssp_exit ( ) ;
} ;
struct platform_driver jornadassp_driver = {
. probe = jornada_ssp_probe ,
2023-03-14 13:32:25 +03:00
. remove_new = jornada_ssp_remove ,
2007-07-20 21:22:57 +04:00
. driver = {
. name = " jornada_ssp " ,
} ,
} ;
static int __init jornada_ssp_init ( void )
{
return platform_driver_register ( & jornadassp_driver ) ;
}
2012-01-20 10:54:51 +04:00
module_init ( jornada_ssp_init ) ;