2013-04-11 03:03:29 +02:00
/*
*
* Intel Management Engine Interface ( Intel MEI ) Linux driver
* Copyright ( c ) 2003 - 2013 , Intel Corporation .
*
* 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 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 .
*
*/
# include <linux/kernel.h>
2013-04-11 03:03:31 +02:00
# include <linux/sched.h>
2013-04-11 03:03:29 +02:00
# include <linux/module.h>
# include <linux/moduleparam.h>
# include <linux/device.h>
# include <linux/pci.h>
# include <linux/mei_cl_bus.h>
# include "mei_dev.h"
# include "client.h"
struct mei_nfc_cmd {
u8 command ;
u8 status ;
u16 req_id ;
u32 reserved ;
u16 data_size ;
u8 sub_command ;
u8 data [ ] ;
} __packed ;
struct mei_nfc_reply {
u8 command ;
u8 status ;
u16 req_id ;
u32 reserved ;
u16 data_size ;
u8 sub_command ;
u8 reply_status ;
u8 data [ ] ;
} __packed ;
struct mei_nfc_if_version {
u8 radio_version_sw [ 3 ] ;
u8 reserved [ 3 ] ;
u8 radio_version_hw [ 3 ] ;
u8 i2c_addr ;
u8 fw_ivn ;
u8 vendor_id ;
u8 radio_type ;
} __packed ;
struct mei_nfc_connect {
u8 fw_ivn ;
u8 vendor_id ;
} __packed ;
struct mei_nfc_connect_resp {
u8 fw_ivn ;
u8 vendor_id ;
u16 me_major ;
u16 me_minor ;
u16 me_hotfix ;
u16 me_build ;
} __packed ;
struct mei_nfc_hci_hdr {
u8 cmd ;
u8 status ;
u16 req_id ;
u32 reserved ;
u16 data_size ;
} __packed ;
# define MEI_NFC_CMD_MAINTENANCE 0x00
# define MEI_NFC_CMD_HCI_SEND 0x01
# define MEI_NFC_CMD_HCI_RECV 0x02
# define MEI_NFC_SUBCMD_CONNECT 0x00
# define MEI_NFC_SUBCMD_IF_VERSION 0x01
# define MEI_NFC_HEADER_SIZE 10
/** mei_nfc_dev - NFC mei device
*
* @ cl : NFC host client
* @ cl_info : NFC info host client
* @ init_work : perform connection to the info client
* @ fw_ivn : NFC Intervace Version Number
* @ vendor_id : NFC manufacturer ID
* @ radio_type : NFC radio type
*/
struct mei_nfc_dev {
struct mei_cl * cl ;
struct mei_cl * cl_info ;
struct work_struct init_work ;
2013-04-11 03:03:31 +02:00
wait_queue_head_t send_wq ;
2013-04-11 03:03:29 +02:00
u8 fw_ivn ;
u8 vendor_id ;
u8 radio_type ;
2013-04-11 03:03:30 +02:00
char * bus_name ;
2013-04-11 03:03:31 +02:00
u16 req_id ;
u16 recv_req_id ;
2013-04-11 03:03:29 +02:00
} ;
static struct mei_nfc_dev nfc_dev ;
/* UUIDs for NFC F/W clients */
const uuid_le mei_nfc_guid = UUID_LE ( 0x0bb17a78 , 0x2a8e , 0x4c50 ,
0x94 , 0xd4 , 0x50 , 0x26 ,
0x67 , 0x23 , 0x77 , 0x5c ) ;
static const uuid_le mei_nfc_info_guid = UUID_LE ( 0xd2de1625 , 0x382d , 0x417d ,
0x48 , 0xa4 , 0xef , 0xab ,
0xba , 0x8a , 0x12 , 0x06 ) ;
2013-04-11 03:03:30 +02:00
/* Vendors */
# define MEI_NFC_VENDOR_INSIDE 0x00
# define MEI_NFC_VENDOR_NXP 0x01
/* Radio types */
# define MEI_NFC_VENDOR_INSIDE_UREAD 0x00
# define MEI_NFC_VENDOR_NXP_PN544 0x01
2013-04-11 03:03:29 +02:00
static void mei_nfc_free ( struct mei_nfc_dev * ndev )
{
if ( ndev - > cl ) {
list_del ( & ndev - > cl - > device_link ) ;
mei_cl_unlink ( ndev - > cl ) ;
kfree ( ndev - > cl ) ;
}
if ( ndev - > cl_info ) {
list_del ( & ndev - > cl_info - > device_link ) ;
mei_cl_unlink ( ndev - > cl_info ) ;
kfree ( ndev - > cl_info ) ;
}
2013-06-10 10:10:26 +03:00
memset ( ndev , 0 , sizeof ( struct mei_nfc_dev ) ) ;
2013-04-11 03:03:29 +02:00
}
2013-04-11 03:03:30 +02:00
static int mei_nfc_build_bus_name ( struct mei_nfc_dev * ndev )
{
struct mei_device * dev ;
if ( ! ndev - > cl )
return - ENODEV ;
dev = ndev - > cl - > dev ;
switch ( ndev - > vendor_id ) {
case MEI_NFC_VENDOR_INSIDE :
switch ( ndev - > radio_type ) {
case MEI_NFC_VENDOR_INSIDE_UREAD :
ndev - > bus_name = " microread " ;
return 0 ;
default :
dev_err ( & dev - > pdev - > dev , " Unknow radio type 0x%x \n " ,
ndev - > radio_type ) ;
return - EINVAL ;
}
case MEI_NFC_VENDOR_NXP :
switch ( ndev - > radio_type ) {
case MEI_NFC_VENDOR_NXP_PN544 :
ndev - > bus_name = " pn544 " ;
return 0 ;
default :
dev_err ( & dev - > pdev - > dev , " Unknow radio type 0x%x \n " ,
ndev - > radio_type ) ;
return - EINVAL ;
}
default :
dev_err ( & dev - > pdev - > dev , " Unknow vendor ID 0x%x \n " ,
ndev - > vendor_id ) ;
return - EINVAL ;
}
return 0 ;
}
2013-04-11 03:03:31 +02:00
static int mei_nfc_connect ( struct mei_nfc_dev * ndev )
{
struct mei_device * dev ;
struct mei_cl * cl ;
struct mei_nfc_cmd * cmd , * reply ;
struct mei_nfc_connect * connect ;
struct mei_nfc_connect_resp * connect_resp ;
size_t connect_length , connect_resp_length ;
int bytes_recv , ret ;
cl = ndev - > cl ;
dev = cl - > dev ;
connect_length = sizeof ( struct mei_nfc_cmd ) +
sizeof ( struct mei_nfc_connect ) ;
connect_resp_length = sizeof ( struct mei_nfc_cmd ) +
sizeof ( struct mei_nfc_connect_resp ) ;
cmd = kzalloc ( connect_length , GFP_KERNEL ) ;
if ( ! cmd )
return - ENOMEM ;
connect = ( struct mei_nfc_connect * ) cmd - > data ;
reply = kzalloc ( connect_resp_length , GFP_KERNEL ) ;
if ( ! reply ) {
kfree ( cmd ) ;
return - ENOMEM ;
}
connect_resp = ( struct mei_nfc_connect_resp * ) reply - > data ;
cmd - > command = MEI_NFC_CMD_MAINTENANCE ;
cmd - > data_size = 3 ;
cmd - > sub_command = MEI_NFC_SUBCMD_CONNECT ;
connect - > fw_ivn = ndev - > fw_ivn ;
connect - > vendor_id = ndev - > vendor_id ;
ret = __mei_cl_send ( cl , ( u8 * ) cmd , connect_length ) ;
if ( ret < 0 ) {
dev_err ( & dev - > pdev - > dev , " Could not send connect cmd \n " ) ;
goto err ;
}
bytes_recv = __mei_cl_recv ( cl , ( u8 * ) reply , connect_resp_length ) ;
if ( bytes_recv < 0 ) {
dev_err ( & dev - > pdev - > dev , " Could not read connect response \n " ) ;
ret = bytes_recv ;
goto err ;
}
dev_info ( & dev - > pdev - > dev , " IVN 0x%x Vendor ID 0x%x \n " ,
connect_resp - > fw_ivn , connect_resp - > vendor_id ) ;
dev_info ( & dev - > pdev - > dev , " ME FW %d.%d.%d.%d \n " ,
connect_resp - > me_major , connect_resp - > me_minor ,
connect_resp - > me_hotfix , connect_resp - > me_build ) ;
ret = 0 ;
err :
kfree ( reply ) ;
kfree ( cmd ) ;
return ret ;
}
2013-04-11 03:03:29 +02:00
static int mei_nfc_if_version ( struct mei_nfc_dev * ndev )
{
struct mei_device * dev ;
struct mei_cl * cl ;
struct mei_nfc_cmd cmd ;
struct mei_nfc_reply * reply = NULL ;
struct mei_nfc_if_version * version ;
size_t if_version_length ;
int bytes_recv , ret ;
cl = ndev - > cl_info ;
dev = cl - > dev ;
memset ( & cmd , 0 , sizeof ( struct mei_nfc_cmd ) ) ;
cmd . command = MEI_NFC_CMD_MAINTENANCE ;
cmd . data_size = 1 ;
cmd . sub_command = MEI_NFC_SUBCMD_IF_VERSION ;
ret = __mei_cl_send ( cl , ( u8 * ) & cmd , sizeof ( struct mei_nfc_cmd ) ) ;
if ( ret < 0 ) {
dev_err ( & dev - > pdev - > dev , " Could not send IF version cmd \n " ) ;
return ret ;
}
/* to be sure on the stack we alloc memory */
if_version_length = sizeof ( struct mei_nfc_reply ) +
sizeof ( struct mei_nfc_if_version ) ;
reply = kzalloc ( if_version_length , GFP_KERNEL ) ;
if ( ! reply )
return - ENOMEM ;
bytes_recv = __mei_cl_recv ( cl , ( u8 * ) reply , if_version_length ) ;
if ( bytes_recv < 0 | | bytes_recv < sizeof ( struct mei_nfc_reply ) ) {
dev_err ( & dev - > pdev - > dev , " Could not read IF version \n " ) ;
ret = - EIO ;
goto err ;
}
version = ( struct mei_nfc_if_version * ) reply - > data ;
ndev - > fw_ivn = version - > fw_ivn ;
ndev - > vendor_id = version - > vendor_id ;
ndev - > radio_type = version - > radio_type ;
err :
kfree ( reply ) ;
return ret ;
}
2013-04-11 03:03:31 +02:00
static int mei_nfc_enable ( struct mei_cl_device * cldev )
{
struct mei_device * dev ;
struct mei_nfc_dev * ndev = & nfc_dev ;
int ret ;
dev = ndev - > cl - > dev ;
ret = mei_nfc_connect ( ndev ) ;
if ( ret < 0 ) {
dev_err ( & dev - > pdev - > dev , " Could not connect to NFC " ) ;
return ret ;
}
return 0 ;
}
static int mei_nfc_disable ( struct mei_cl_device * cldev )
{
return 0 ;
}
static int mei_nfc_send ( struct mei_cl_device * cldev , u8 * buf , size_t length )
{
struct mei_device * dev ;
struct mei_nfc_dev * ndev ;
struct mei_nfc_hci_hdr * hdr ;
u8 * mei_buf ;
int err ;
ndev = ( struct mei_nfc_dev * ) cldev - > priv_data ;
dev = ndev - > cl - > dev ;
mei_buf = kzalloc ( length + MEI_NFC_HEADER_SIZE , GFP_KERNEL ) ;
if ( ! mei_buf )
return - ENOMEM ;
hdr = ( struct mei_nfc_hci_hdr * ) mei_buf ;
hdr - > cmd = MEI_NFC_CMD_HCI_SEND ;
hdr - > status = 0 ;
hdr - > req_id = ndev - > req_id ;
hdr - > reserved = 0 ;
hdr - > data_size = length ;
memcpy ( mei_buf + MEI_NFC_HEADER_SIZE , buf , length ) ;
err = __mei_cl_send ( ndev - > cl , mei_buf , length + MEI_NFC_HEADER_SIZE ) ;
if ( err < 0 )
return err ;
kfree ( mei_buf ) ;
if ( ! wait_event_interruptible_timeout ( ndev - > send_wq ,
ndev - > recv_req_id = = ndev - > req_id , HZ ) ) {
dev_err ( & dev - > pdev - > dev , " NFC MEI command timeout \n " ) ;
err = - ETIMEDOUT ;
} else {
ndev - > req_id + + ;
}
return err ;
}
static int mei_nfc_recv ( struct mei_cl_device * cldev , u8 * buf , size_t length )
{
struct mei_nfc_dev * ndev ;
struct mei_nfc_hci_hdr * hci_hdr ;
int received_length ;
ndev = ( struct mei_nfc_dev * ) cldev - > priv_data ;
received_length = __mei_cl_recv ( ndev - > cl , buf , length ) ;
if ( received_length < 0 )
return received_length ;
hci_hdr = ( struct mei_nfc_hci_hdr * ) buf ;
if ( hci_hdr - > cmd = = MEI_NFC_CMD_HCI_SEND ) {
ndev - > recv_req_id = hci_hdr - > req_id ;
wake_up ( & ndev - > send_wq ) ;
return 0 ;
}
return received_length ;
}
static struct mei_cl_ops nfc_ops = {
. enable = mei_nfc_enable ,
. disable = mei_nfc_disable ,
. send = mei_nfc_send ,
. recv = mei_nfc_recv ,
} ;
2013-04-11 03:03:29 +02:00
static void mei_nfc_init ( struct work_struct * work )
{
struct mei_device * dev ;
2013-04-11 03:03:30 +02:00
struct mei_cl_device * cldev ;
2013-04-11 03:03:29 +02:00
struct mei_nfc_dev * ndev ;
struct mei_cl * cl_info ;
ndev = container_of ( work , struct mei_nfc_dev , init_work ) ;
cl_info = ndev - > cl_info ;
dev = cl_info - > dev ;
mutex_lock ( & dev - > device_lock ) ;
if ( mei_cl_connect ( cl_info , NULL ) < 0 ) {
mutex_unlock ( & dev - > device_lock ) ;
dev_err ( & dev - > pdev - > dev ,
" Could not connect to the NFC INFO ME client " ) ;
goto err ;
}
mutex_unlock ( & dev - > device_lock ) ;
if ( mei_nfc_if_version ( ndev ) < 0 ) {
dev_err ( & dev - > pdev - > dev , " Could not get the NFC interfave version " ) ;
goto err ;
}
dev_info ( & dev - > pdev - > dev ,
" NFC MEI VERSION: IVN 0x%x Vendor ID 0x%x Type 0x%x \n " ,
ndev - > fw_ivn , ndev - > vendor_id , ndev - > radio_type ) ;
mutex_lock ( & dev - > device_lock ) ;
if ( mei_cl_disconnect ( cl_info ) < 0 ) {
mutex_unlock ( & dev - > device_lock ) ;
dev_err ( & dev - > pdev - > dev ,
" Could not disconnect the NFC INFO ME client " ) ;
goto err ;
}
mutex_unlock ( & dev - > device_lock ) ;
2013-04-11 03:03:30 +02:00
if ( mei_nfc_build_bus_name ( ndev ) < 0 ) {
dev_err ( & dev - > pdev - > dev ,
" Could not build the bus ID name \n " ) ;
return ;
}
2013-04-11 03:03:31 +02:00
cldev = mei_cl_add_device ( dev , mei_nfc_guid , ndev - > bus_name , & nfc_ops ) ;
2013-04-11 03:03:30 +02:00
if ( ! cldev ) {
dev_err ( & dev - > pdev - > dev ,
" Could not add the NFC device to the MEI bus \n " ) ;
goto err ;
}
cldev - > priv_data = ndev ;
2013-04-11 03:03:29 +02:00
return ;
err :
mei_nfc_free ( ndev ) ;
return ;
}
int mei_nfc_host_init ( struct mei_device * dev )
{
struct mei_nfc_dev * ndev = & nfc_dev ;
struct mei_cl * cl_info , * cl = NULL ;
int i , ret ;
/* already initialzed */
if ( ndev - > cl_info )
return 0 ;
cl_info = mei_cl_allocate ( dev ) ;
cl = mei_cl_allocate ( dev ) ;
if ( ! cl | | ! cl_info ) {
ret = - ENOMEM ;
goto err ;
}
/* check for valid client id */
i = mei_me_cl_by_uuid ( dev , & mei_nfc_info_guid ) ;
if ( i < 0 ) {
dev_info ( & dev - > pdev - > dev , " nfc: failed to find the client \n " ) ;
ret = - ENOENT ;
goto err ;
}
cl_info - > me_client_id = dev - > me_clients [ i ] . client_id ;
ret = mei_cl_link ( cl_info , MEI_HOST_CLIENT_ID_ANY ) ;
if ( ret )
goto err ;
cl_info - > device_uuid = mei_nfc_info_guid ;
list_add_tail ( & cl_info - > device_link , & dev - > device_list ) ;
/* check for valid client id */
i = mei_me_cl_by_uuid ( dev , & mei_nfc_guid ) ;
if ( i < 0 ) {
dev_info ( & dev - > pdev - > dev , " nfc: failed to find the client \n " ) ;
ret = - ENOENT ;
goto err ;
}
cl - > me_client_id = dev - > me_clients [ i ] . client_id ;
ret = mei_cl_link ( cl , MEI_HOST_CLIENT_ID_ANY ) ;
if ( ret )
goto err ;
cl - > device_uuid = mei_nfc_guid ;
list_add_tail ( & cl - > device_link , & dev - > device_list ) ;
ndev - > cl_info = cl_info ;
ndev - > cl = cl ;
2013-04-11 03:03:31 +02:00
ndev - > req_id = 1 ;
2013-04-11 03:03:29 +02:00
INIT_WORK ( & ndev - > init_work , mei_nfc_init ) ;
2013-04-11 03:03:31 +02:00
init_waitqueue_head ( & ndev - > send_wq ) ;
2013-04-11 03:03:29 +02:00
schedule_work ( & ndev - > init_work ) ;
return 0 ;
err :
mei_nfc_free ( ndev ) ;
return ret ;
}
void mei_nfc_host_exit ( void )
{
struct mei_nfc_dev * ndev = & nfc_dev ;
2013-04-11 03:03:30 +02:00
if ( ndev - > cl & & ndev - > cl - > device )
mei_cl_remove_device ( ndev - > cl - > device ) ;
2013-04-11 03:03:29 +02:00
mei_nfc_free ( ndev ) ;
}