2015-08-14 22:33:37 +02:00
/*
* SPI Link Layer for ST NCI based Driver
* Copyright ( C ) 2014 - 2015 STMicroelectronics SAS . All rights reserved .
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms and conditions of the GNU General Public License ,
* version 2 , as published by the Free Software Foundation .
*
* 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 , see < http : //www.gnu.org/licenses/>.
*/
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
# include <linux/module.h>
# include <linux/spi/spi.h>
nfc: st-nci: Add support for acpi probing for spi device.
Add support for acpi probing.
SMO2101 is used for st21nfcb
It has been tested with the following acpi node on Minnowboard:
Note: Remove uicc-present or ese-present Package if one of them is not
supported.
Device (NFC1)
{
Name (_ADR, Zero) // _ADR: Address
Name (_HID, "SMO2101") // _HID: Hardware ID
Name (_CID, "SMO2101") // _CID: Compatible ID
Name (_DDN, "SMO NFC") // _DDN: DOS Device Name
Name (_UID, One) // _UID: Unique ID
Name (_DSD, Package (0x02)
{
/* Device Properties for _DSD */
ToUUID ("daffd814-6eba-4d8c-8a91-bc9bbf4aa301"),
Package (0x02)
{
Package (0x02) { "uicc-present", 1 },
Package (0x02) { "ese-present", 1 }
}
})
Method (_CRS, 0, NotSerialized) // _CRS: Current Resource Settings
{
Name (SBUF, ResourceTemplate ()
{
SpiSerialBus (0, PolarityLow, FourWireMode, 8,
ControllerInitiated, 4000000, ClockPolarityLow,
ClockPhaseFirst, "\\_SB.SPI1",
0x00, ResourceConsumer, ,)
GpioInt (Edge, ActiveHigh, ExclusiveAndWake, PullNone, 0x0000,
"\\_SB.GPO2", 0x00, ResourceConsumer, ,)
{ // Pin list
0x0001
}
GpioIo (Exclusive, PullDefault, 0x0000, 0x0000, IoRestrictionOutputOnly,
"\\_SB.GPO2", 0x00, ResourceConsumer, ,)
{ // Pin list
0x0002,
}
})
Return (SBUF) /* \_SB_.SPI1.NFC1._CRS.SBUF */
}
Method (_STA, 0, NotSerialized) // _STA: Status
{
Return (0x0F)
}
}
Signed-off-by: Christophe Ricard <christophe-h.ricard@st.com>
Signed-off-by: Samuel Ortiz <sameo@linux.intel.com>
2015-12-23 23:45:10 +01:00
# include <linux/gpio/consumer.h>
# include <linux/acpi.h>
2015-08-14 22:33:37 +02:00
# include <linux/interrupt.h>
# include <linux/delay.h>
# include <linux/nfc.h>
2017-06-19 13:08:54 +03:00
# include <linux/of.h>
2015-10-25 22:54:19 +01:00
# include <net/nfc/nci.h>
2015-08-14 22:33:37 +02:00
2015-10-25 22:54:17 +01:00
# include "st-nci.h"
2015-08-14 22:33:37 +02:00
# define DRIVER_DESC "NCI NFC driver for ST_NCI"
/* ndlc header */
# define ST_NCI_FRAME_HEADROOM 1
2015-12-23 23:45:12 +01:00
# define ST_NCI_FRAME_TAILROOM 0
2015-08-14 22:33:37 +02:00
# define ST_NCI_SPI_MIN_SIZE 4 /* PCB(1) + NCI Packet header(3) */
# define ST_NCI_SPI_MAX_SIZE 250 /* req 4.2.1 */
2017-06-19 13:08:52 +03:00
# define ST_NCI_DRIVER_NAME "st_nci"
2015-08-14 22:33:37 +02:00
# define ST_NCI_SPI_DRIVER_NAME "st_nci_spi"
struct st_nci_spi_phy {
struct spi_device * spi_dev ;
struct llt_ndlc * ndlc ;
2015-10-25 22:54:48 +01:00
bool irq_active ;
2017-06-19 13:08:54 +03:00
struct gpio_desc * gpiod_reset ;
2015-10-25 22:54:39 +01:00
struct st_nci_se_status se_status ;
2015-08-14 22:33:37 +02:00
} ;
static int st_nci_spi_enable ( void * phy_id )
{
struct st_nci_spi_phy * phy = phy_id ;
2017-06-19 13:08:54 +03:00
gpiod_set_value ( phy - > gpiod_reset , 0 ) ;
2015-08-14 22:33:37 +02:00
usleep_range ( 10000 , 15000 ) ;
2017-06-19 13:08:54 +03:00
gpiod_set_value ( phy - > gpiod_reset , 1 ) ;
2015-08-14 22:33:37 +02:00
usleep_range ( 80000 , 85000 ) ;
2015-10-25 22:54:48 +01:00
if ( phy - > ndlc - > powered = = 0 & & phy - > irq_active = = 0 ) {
2015-08-14 22:33:37 +02:00
enable_irq ( phy - > spi_dev - > irq ) ;
2015-10-25 22:54:48 +01:00
phy - > irq_active = true ;
}
2015-08-14 22:33:37 +02:00
return 0 ;
}
static void st_nci_spi_disable ( void * phy_id )
{
struct st_nci_spi_phy * phy = phy_id ;
disable_irq_nosync ( phy - > spi_dev - > irq ) ;
2015-10-25 22:54:48 +01:00
phy - > irq_active = false ;
2015-08-14 22:33:37 +02:00
}
/*
* Writing a frame must not return the number of written bytes .
* It must return either zero for success , or < 0 for error .
* In addition , it must not alter the skb
*/
static int st_nci_spi_write ( void * phy_id , struct sk_buff * skb )
{
int r ;
struct st_nci_spi_phy * phy = phy_id ;
struct spi_device * dev = phy - > spi_dev ;
struct sk_buff * skb_rx ;
2015-10-25 22:54:19 +01:00
u8 buf [ ST_NCI_SPI_MAX_SIZE + NCI_DATA_HDR_SIZE +
ST_NCI_FRAME_HEADROOM + ST_NCI_FRAME_TAILROOM ] ;
2015-08-14 22:33:37 +02:00
struct spi_transfer spi_xfer = {
. tx_buf = skb - > data ,
. rx_buf = buf ,
. len = skb - > len ,
} ;
if ( phy - > ndlc - > hard_fault ! = 0 )
return phy - > ndlc - > hard_fault ;
r = spi_sync_transfer ( dev , & spi_xfer , 1 ) ;
/*
* We may have received some valuable data on miso line .
* Send them back in the ndlc state machine .
*/
if ( ! r ) {
skb_rx = alloc_skb ( skb - > len , GFP_KERNEL ) ;
if ( ! skb_rx ) {
r = - ENOMEM ;
goto exit ;
}
skb_put ( skb_rx , skb - > len ) ;
memcpy ( skb_rx - > data , buf , skb - > len ) ;
ndlc_recv ( phy - > ndlc , skb_rx ) ;
}
exit :
return r ;
}
/*
* Reads an ndlc frame and returns it in a newly allocated sk_buff .
* returns :
* 0 : if received frame is complete
* - EREMOTEIO : i2c read error ( fatal )
* - EBADMSG : frame was incorrect and discarded
* - ENOMEM : cannot allocate skb , frame dropped
*/
static int st_nci_spi_read ( struct st_nci_spi_phy * phy ,
struct sk_buff * * skb )
{
int r ;
u8 len ;
u8 buf [ ST_NCI_SPI_MAX_SIZE ] ;
struct spi_device * dev = phy - > spi_dev ;
struct spi_transfer spi_xfer = {
. rx_buf = buf ,
. len = ST_NCI_SPI_MIN_SIZE ,
} ;
r = spi_sync_transfer ( dev , & spi_xfer , 1 ) ;
if ( r < 0 )
return - EREMOTEIO ;
len = be16_to_cpu ( * ( __be16 * ) ( buf + 2 ) ) ;
if ( len > ST_NCI_SPI_MAX_SIZE ) {
nfc_err ( & dev - > dev , " invalid frame len \n " ) ;
phy - > ndlc - > hard_fault = 1 ;
return - EBADMSG ;
}
* skb = alloc_skb ( ST_NCI_SPI_MIN_SIZE + len , GFP_KERNEL ) ;
if ( * skb = = NULL )
return - ENOMEM ;
skb_reserve ( * skb , ST_NCI_SPI_MIN_SIZE ) ;
skb_put ( * skb , ST_NCI_SPI_MIN_SIZE ) ;
memcpy ( ( * skb ) - > data , buf , ST_NCI_SPI_MIN_SIZE ) ;
if ( ! len )
return 0 ;
spi_xfer . len = len ;
r = spi_sync_transfer ( dev , & spi_xfer , 1 ) ;
if ( r < 0 ) {
kfree_skb ( * skb ) ;
return - EREMOTEIO ;
}
skb_put ( * skb , len ) ;
memcpy ( ( * skb ) - > data + ST_NCI_SPI_MIN_SIZE , buf , len ) ;
return 0 ;
}
/*
* Reads an ndlc frame from the chip .
*
* On ST21NFCB , IRQ goes in idle state when read starts .
*/
static irqreturn_t st_nci_irq_thread_fn ( int irq , void * phy_id )
{
struct st_nci_spi_phy * phy = phy_id ;
struct spi_device * dev ;
struct sk_buff * skb = NULL ;
int r ;
if ( ! phy | | ! phy - > ndlc | | irq ! = phy - > spi_dev - > irq ) {
WARN_ON_ONCE ( 1 ) ;
return IRQ_NONE ;
}
dev = phy - > spi_dev ;
dev_dbg ( & dev - > dev , " IRQ \n " ) ;
if ( phy - > ndlc - > hard_fault )
return IRQ_HANDLED ;
if ( ! phy - > ndlc - > powered ) {
st_nci_spi_disable ( phy ) ;
return IRQ_HANDLED ;
}
r = st_nci_spi_read ( phy , & skb ) ;
if ( r = = - EREMOTEIO | | r = = - ENOMEM | | r = = - EBADMSG )
return IRQ_HANDLED ;
ndlc_recv ( phy - > ndlc , skb ) ;
return IRQ_HANDLED ;
}
static struct nfc_phy_ops spi_phy_ops = {
. write = st_nci_spi_write ,
. enable = st_nci_spi_enable ,
. disable = st_nci_spi_disable ,
} ;
2017-06-19 13:08:56 +03:00
static const struct acpi_gpio_params reset_gpios = { 1 , 0 , false } ;
static const struct acpi_gpio_mapping acpi_st_nci_gpios [ ] = {
{ " reset-gpios " , & reset_gpios , 1 } ,
{ } ,
} ;
2015-08-14 22:33:37 +02:00
static int st_nci_spi_probe ( struct spi_device * dev )
{
struct st_nci_spi_phy * phy ;
int r ;
dev_dbg ( & dev - > dev , " %s \n " , __func__ ) ;
dev_dbg ( & dev - > dev , " IRQ: %d \n " , dev - > irq ) ;
/* Check SPI platform functionnalities */
if ( ! dev ) {
pr_debug ( " %s: dev is NULL. Device is not accessible. \n " ,
__func__ ) ;
return - ENODEV ;
}
phy = devm_kzalloc ( & dev - > dev , sizeof ( struct st_nci_spi_phy ) ,
GFP_KERNEL ) ;
if ( ! phy )
return - ENOMEM ;
phy - > spi_dev = dev ;
spi_set_drvdata ( dev , phy ) ;
2017-06-19 13:08:57 +03:00
r = devm_acpi_dev_add_driver_gpios ( & dev - > dev , acpi_st_nci_gpios ) ;
if ( r )
dev_dbg ( & dev - > dev , " Unable to add GPIO mapping table \n " ) ;
/* Get RESET GPIO */
phy - > gpiod_reset = devm_gpiod_get ( & dev - > dev , " reset " , GPIOD_OUT_HIGH ) ;
if ( IS_ERR ( phy - > gpiod_reset ) ) {
nfc_err ( & dev - > dev , " Unable to get RESET GPIO \n " ) ;
return PTR_ERR ( phy - > gpiod_reset ) ;
2015-08-14 22:33:37 +02:00
}
2017-06-19 13:08:55 +03:00
phy - > se_status . is_ese_present =
device_property_read_bool ( & dev - > dev , " ese-present " ) ;
phy - > se_status . is_uicc_present =
device_property_read_bool ( & dev - > dev , " uicc-present " ) ;
2015-08-14 22:33:37 +02:00
r = ndlc_probe ( phy , & spi_phy_ops , & dev - > dev ,
ST_NCI_FRAME_HEADROOM , ST_NCI_FRAME_TAILROOM ,
2015-10-25 22:54:39 +01:00
& phy - > ndlc , & phy - > se_status ) ;
2015-08-14 22:33:37 +02:00
if ( r < 0 ) {
nfc_err ( & dev - > dev , " Unable to register ndlc layer \n " ) ;
return r ;
}
2015-10-25 22:54:48 +01:00
phy - > irq_active = true ;
2015-08-14 22:33:37 +02:00
r = devm_request_threaded_irq ( & dev - > dev , dev - > irq , NULL ,
st_nci_irq_thread_fn ,
2017-06-19 13:08:53 +03:00
IRQF_ONESHOT ,
2015-08-14 22:33:37 +02:00
ST_NCI_SPI_DRIVER_NAME , phy ) ;
if ( r < 0 )
nfc_err ( & dev - > dev , " Unable to register IRQ handler \n " ) ;
return r ;
}
static int st_nci_spi_remove ( struct spi_device * dev )
{
struct st_nci_spi_phy * phy = spi_get_drvdata ( dev ) ;
dev_dbg ( & dev - > dev , " %s \n " , __func__ ) ;
ndlc_remove ( phy - > ndlc ) ;
return 0 ;
}
2015-12-23 23:45:05 +01:00
static struct spi_device_id st_nci_spi_id_table [ ] = {
{ ST_NCI_SPI_DRIVER_NAME , 0 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( spi , st_nci_spi_id_table ) ;
nfc: st-nci: Add support for acpi probing for spi device.
Add support for acpi probing.
SMO2101 is used for st21nfcb
It has been tested with the following acpi node on Minnowboard:
Note: Remove uicc-present or ese-present Package if one of them is not
supported.
Device (NFC1)
{
Name (_ADR, Zero) // _ADR: Address
Name (_HID, "SMO2101") // _HID: Hardware ID
Name (_CID, "SMO2101") // _CID: Compatible ID
Name (_DDN, "SMO NFC") // _DDN: DOS Device Name
Name (_UID, One) // _UID: Unique ID
Name (_DSD, Package (0x02)
{
/* Device Properties for _DSD */
ToUUID ("daffd814-6eba-4d8c-8a91-bc9bbf4aa301"),
Package (0x02)
{
Package (0x02) { "uicc-present", 1 },
Package (0x02) { "ese-present", 1 }
}
})
Method (_CRS, 0, NotSerialized) // _CRS: Current Resource Settings
{
Name (SBUF, ResourceTemplate ()
{
SpiSerialBus (0, PolarityLow, FourWireMode, 8,
ControllerInitiated, 4000000, ClockPolarityLow,
ClockPhaseFirst, "\\_SB.SPI1",
0x00, ResourceConsumer, ,)
GpioInt (Edge, ActiveHigh, ExclusiveAndWake, PullNone, 0x0000,
"\\_SB.GPO2", 0x00, ResourceConsumer, ,)
{ // Pin list
0x0001
}
GpioIo (Exclusive, PullDefault, 0x0000, 0x0000, IoRestrictionOutputOnly,
"\\_SB.GPO2", 0x00, ResourceConsumer, ,)
{ // Pin list
0x0002,
}
})
Return (SBUF) /* \_SB_.SPI1.NFC1._CRS.SBUF */
}
Method (_STA, 0, NotSerialized) // _STA: Status
{
Return (0x0F)
}
}
Signed-off-by: Christophe Ricard <christophe-h.ricard@st.com>
Signed-off-by: Samuel Ortiz <sameo@linux.intel.com>
2015-12-23 23:45:10 +01:00
static const struct acpi_device_id st_nci_spi_acpi_match [ ] = {
{ " SMO2101 " , 0 } ,
{ }
} ;
MODULE_DEVICE_TABLE ( acpi , st_nci_spi_acpi_match ) ;
2015-08-14 22:33:37 +02:00
static const struct of_device_id of_st_nci_spi_match [ ] = {
{ . compatible = " st,st21nfcb-spi " , } ,
{ }
} ;
MODULE_DEVICE_TABLE ( of , of_st_nci_spi_match ) ;
static struct spi_driver st_nci_spi_driver = {
. driver = {
. name = ST_NCI_SPI_DRIVER_NAME ,
. of_match_table = of_match_ptr ( of_st_nci_spi_match ) ,
nfc: st-nci: Add support for acpi probing for spi device.
Add support for acpi probing.
SMO2101 is used for st21nfcb
It has been tested with the following acpi node on Minnowboard:
Note: Remove uicc-present or ese-present Package if one of them is not
supported.
Device (NFC1)
{
Name (_ADR, Zero) // _ADR: Address
Name (_HID, "SMO2101") // _HID: Hardware ID
Name (_CID, "SMO2101") // _CID: Compatible ID
Name (_DDN, "SMO NFC") // _DDN: DOS Device Name
Name (_UID, One) // _UID: Unique ID
Name (_DSD, Package (0x02)
{
/* Device Properties for _DSD */
ToUUID ("daffd814-6eba-4d8c-8a91-bc9bbf4aa301"),
Package (0x02)
{
Package (0x02) { "uicc-present", 1 },
Package (0x02) { "ese-present", 1 }
}
})
Method (_CRS, 0, NotSerialized) // _CRS: Current Resource Settings
{
Name (SBUF, ResourceTemplate ()
{
SpiSerialBus (0, PolarityLow, FourWireMode, 8,
ControllerInitiated, 4000000, ClockPolarityLow,
ClockPhaseFirst, "\\_SB.SPI1",
0x00, ResourceConsumer, ,)
GpioInt (Edge, ActiveHigh, ExclusiveAndWake, PullNone, 0x0000,
"\\_SB.GPO2", 0x00, ResourceConsumer, ,)
{ // Pin list
0x0001
}
GpioIo (Exclusive, PullDefault, 0x0000, 0x0000, IoRestrictionOutputOnly,
"\\_SB.GPO2", 0x00, ResourceConsumer, ,)
{ // Pin list
0x0002,
}
})
Return (SBUF) /* \_SB_.SPI1.NFC1._CRS.SBUF */
}
Method (_STA, 0, NotSerialized) // _STA: Status
{
Return (0x0F)
}
}
Signed-off-by: Christophe Ricard <christophe-h.ricard@st.com>
Signed-off-by: Samuel Ortiz <sameo@linux.intel.com>
2015-12-23 23:45:10 +01:00
. acpi_match_table = ACPI_PTR ( st_nci_spi_acpi_match ) ,
2015-08-14 22:33:37 +02:00
} ,
. probe = st_nci_spi_probe ,
. id_table = st_nci_spi_id_table ,
. remove = st_nci_spi_remove ,
} ;
module_spi_driver ( st_nci_spi_driver ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( DRIVER_DESC ) ;