Commit cb675f5f authored by John W. Linville's avatar John W. Linville

Merge tag 'nfc-next-3.8-2' of git://git.kernel.org/pub/scm/linux/kernel/git/sameo/nfc-3.0

Samuel says:

"This is the 2nd NFC pull request for 3.8.

With this one we have:

- A few HCI improvements in preparation for an upcoming HCI chipset support.
- A pn544 code cleanup after the old driver was removed.
- An LLCP improvement for notifying user space when one peer stops ACKing I
  frames."
Signed-off-by: default avatarJohn W. Linville <linville@tuxdriver.com>
parents ac46ba43 be02b6b6
...@@ -5067,6 +5067,7 @@ F: net/nfc/ ...@@ -5067,6 +5067,7 @@ F: net/nfc/
F: include/linux/nfc.h F: include/linux/nfc.h
F: include/net/nfc/ F: include/net/nfc/
F: drivers/nfc/ F: drivers/nfc/
F: include/linux/platform_data/pn544.h
NFS, SUNRPC, AND LOCKD CLIENTS NFS, SUNRPC, AND LOCKD CLIENTS
M: Trond Myklebust <Trond.Myklebust@netapp.com> M: Trond Myklebust <Trond.Myklebust@netapp.com>
......
...@@ -26,7 +26,7 @@ ...@@ -26,7 +26,7 @@
#include <linux/interrupt.h> #include <linux/interrupt.h>
#include <linux/delay.h> #include <linux/delay.h>
#include <linux/nfc/pn544.h> #include <linux/platform_data/pn544.h>
#include <net/nfc/hci.h> #include <net/nfc/hci.h>
#include <net/nfc/llc.h> #include <net/nfc/llc.h>
......
...@@ -25,65 +25,6 @@ ...@@ -25,65 +25,6 @@
#include <linux/i2c.h> #include <linux/i2c.h>
#define PN544_DRIVER_NAME "pn544"
#define PN544_MAXWINDOW_SIZE 7
#define PN544_WINDOW_SIZE 4
#define PN544_RETRIES 10
#define PN544_MAX_I2C_TRANSFER 0x0400
#define PN544_MSG_MAX_SIZE 0x21 /* at normal HCI mode */
/* ioctl */
#define PN544_CHAR_BASE 'P'
#define PN544_IOR(num, dtype) _IOR(PN544_CHAR_BASE, num, dtype)
#define PN544_IOW(num, dtype) _IOW(PN544_CHAR_BASE, num, dtype)
#define PN544_GET_FW_MODE PN544_IOW(1, unsigned int)
#define PN544_SET_FW_MODE PN544_IOW(2, unsigned int)
#define PN544_GET_DEBUG PN544_IOW(3, unsigned int)
#define PN544_SET_DEBUG PN544_IOW(4, unsigned int)
/* Timing restrictions (ms) */
#define PN544_RESETVEN_TIME 30 /* 7 */
#define PN544_PVDDVEN_TIME 0
#define PN544_VBATVEN_TIME 0
#define PN544_GPIO4VEN_TIME 0
#define PN544_WAKEUP_ACK 5
#define PN544_WAKEUP_GUARD (PN544_WAKEUP_ACK + 1)
#define PN544_INACTIVITY_TIME 1000
#define PN544_INTERFRAME_DELAY 200 /* us */
#define PN544_BAUDRATE_CHANGE 150 /* us */
/* Debug bits */
#define PN544_DEBUG_BUF 0x01
#define PN544_DEBUG_READ 0x02
#define PN544_DEBUG_WRITE 0x04
#define PN544_DEBUG_IRQ 0x08
#define PN544_DEBUG_CALLS 0x10
#define PN544_DEBUG_MODE 0x20
/* Normal (HCI) mode */
#define PN544_LLC_HCI_OVERHEAD 3 /* header + crc (to length) */
#define PN544_LLC_MIN_SIZE (1 + PN544_LLC_HCI_OVERHEAD) /* length + */
#define PN544_LLC_MAX_DATA (PN544_MSG_MAX_SIZE - 2)
#define PN544_LLC_MAX_HCI_SIZE (PN544_LLC_MAX_DATA - 2)
struct pn544_llc_packet {
unsigned char length; /* of rest of packet */
unsigned char header;
unsigned char data[PN544_LLC_MAX_DATA]; /* includes crc-ccitt */
};
/* Firmware upgrade mode */
#define PN544_FW_HEADER_SIZE 3
/* max fw transfer is 1024bytes, but I2C limits it to 0xC0 */
#define PN544_MAX_FW_DATA (PN544_MAX_I2C_TRANSFER - PN544_FW_HEADER_SIZE)
struct pn544_fw_packet {
unsigned char command; /* status in answer */
unsigned char length[2]; /* big-endian order (msf) */
unsigned char data[PN544_MAX_FW_DATA];
};
#ifdef __KERNEL__
enum { enum {
NFC_GPIO_ENABLE, NFC_GPIO_ENABLE,
NFC_GPIO_FW_RESET, NFC_GPIO_FW_RESET,
...@@ -99,6 +40,5 @@ struct pn544_nfc_platform_data { ...@@ -99,6 +40,5 @@ struct pn544_nfc_platform_data {
void (*disable) (void); void (*disable) (void);
int (*get_gpio)(int type); int (*get_gpio)(int type);
}; };
#endif /* __KERNEL__ */
#endif /* _PN544_H_ */ #endif /* _PN544_H_ */
...@@ -149,6 +149,8 @@ void *nfc_hci_get_clientdata(struct nfc_hci_dev *hdev); ...@@ -149,6 +149,8 @@ void *nfc_hci_get_clientdata(struct nfc_hci_dev *hdev);
void nfc_hci_driver_failure(struct nfc_hci_dev *hdev, int err); void nfc_hci_driver_failure(struct nfc_hci_dev *hdev, int err);
int nfc_hci_result_to_errno(u8 result);
/* Host IDs */ /* Host IDs */
#define NFC_HCI_HOST_CONTROLLER_ID 0x00 #define NFC_HCI_HOST_CONTROLLER_ID 0x00
#define NFC_HCI_TERMINAL_HOST_ID 0x01 #define NFC_HCI_TERMINAL_HOST_ID 0x01
...@@ -235,5 +237,6 @@ int nfc_hci_send_response(struct nfc_hci_dev *hdev, u8 gate, u8 response, ...@@ -235,5 +237,6 @@ int nfc_hci_send_response(struct nfc_hci_dev *hdev, u8 gate, u8 response,
int nfc_hci_send_event(struct nfc_hci_dev *hdev, u8 gate, u8 event, int nfc_hci_send_event(struct nfc_hci_dev *hdev, u8 gate, u8 event,
const u8 *param, size_t param_len); const u8 *param, size_t param_len);
int nfc_hci_target_discovered(struct nfc_hci_dev *hdev, u8 gate); int nfc_hci_target_discovered(struct nfc_hci_dev *hdev, u8 gate);
u32 nfc_hci_sak_to_protocol(u8 sak);
#endif /* __NET_HCI_H */ #endif /* __NET_HCI_H */
...@@ -344,7 +344,7 @@ int nfc_hci_connect_gate(struct nfc_hci_dev *hdev, u8 dest_host, u8 dest_gate, ...@@ -344,7 +344,7 @@ int nfc_hci_connect_gate(struct nfc_hci_dev *hdev, u8 dest_host, u8 dest_gate,
return -EADDRINUSE; return -EADDRINUSE;
if (pipe != NFC_HCI_INVALID_PIPE) if (pipe != NFC_HCI_INVALID_PIPE)
goto pipe_is_open; goto open_pipe;
switch (dest_gate) { switch (dest_gate) {
case NFC_HCI_LINK_MGMT_GATE: case NFC_HCI_LINK_MGMT_GATE:
...@@ -361,6 +361,7 @@ int nfc_hci_connect_gate(struct nfc_hci_dev *hdev, u8 dest_host, u8 dest_gate, ...@@ -361,6 +361,7 @@ int nfc_hci_connect_gate(struct nfc_hci_dev *hdev, u8 dest_host, u8 dest_gate,
break; break;
} }
open_pipe:
r = nfc_hci_open_pipe(hdev, pipe); r = nfc_hci_open_pipe(hdev, pipe);
if (r < 0) { if (r < 0) {
if (pipe_created) if (pipe_created)
...@@ -371,7 +372,6 @@ int nfc_hci_connect_gate(struct nfc_hci_dev *hdev, u8 dest_host, u8 dest_gate, ...@@ -371,7 +372,6 @@ int nfc_hci_connect_gate(struct nfc_hci_dev *hdev, u8 dest_host, u8 dest_gate,
return r; return r;
} }
pipe_is_open:
hdev->gate2pipe[dest_gate] = pipe; hdev->gate2pipe[dest_gate] = pipe;
return 0; return 0;
......
...@@ -33,17 +33,20 @@ ...@@ -33,17 +33,20 @@
/* Largest headroom needed for outgoing HCI commands */ /* Largest headroom needed for outgoing HCI commands */
#define HCI_CMDS_HEADROOM 1 #define HCI_CMDS_HEADROOM 1
static int nfc_hci_result_to_errno(u8 result) int nfc_hci_result_to_errno(u8 result)
{ {
switch (result) { switch (result) {
case NFC_HCI_ANY_OK: case NFC_HCI_ANY_OK:
return 0; return 0;
case NFC_HCI_ANY_E_REG_PAR_UNKNOWN:
return -EOPNOTSUPP;
case NFC_HCI_ANY_E_TIMEOUT: case NFC_HCI_ANY_E_TIMEOUT:
return -ETIME; return -ETIME;
default: default:
return -1; return -1;
} }
} }
EXPORT_SYMBOL(nfc_hci_result_to_errno);
static void nfc_hci_msg_tx_work(struct work_struct *work) static void nfc_hci_msg_tx_work(struct work_struct *work)
{ {
...@@ -167,7 +170,7 @@ void nfc_hci_cmd_received(struct nfc_hci_dev *hdev, u8 pipe, u8 cmd, ...@@ -167,7 +170,7 @@ void nfc_hci_cmd_received(struct nfc_hci_dev *hdev, u8 pipe, u8 cmd,
kfree_skb(skb); kfree_skb(skb);
} }
static u32 nfc_hci_sak_to_protocol(u8 sak) u32 nfc_hci_sak_to_protocol(u8 sak)
{ {
switch (NFC_HCI_TYPE_A_SEL_PROT(sak)) { switch (NFC_HCI_TYPE_A_SEL_PROT(sak)) {
case NFC_HCI_TYPE_A_SEL_PROT_MIFARE: case NFC_HCI_TYPE_A_SEL_PROT_MIFARE:
...@@ -182,6 +185,7 @@ static u32 nfc_hci_sak_to_protocol(u8 sak) ...@@ -182,6 +185,7 @@ static u32 nfc_hci_sak_to_protocol(u8 sak)
return 0xffffffff; return 0xffffffff;
} }
} }
EXPORT_SYMBOL(nfc_hci_sak_to_protocol);
int nfc_hci_target_discovered(struct nfc_hci_dev *hdev, u8 gate) int nfc_hci_target_discovered(struct nfc_hci_dev *hdev, u8 gate)
{ {
...@@ -284,6 +288,12 @@ void nfc_hci_event_received(struct nfc_hci_dev *hdev, u8 pipe, u8 event, ...@@ -284,6 +288,12 @@ void nfc_hci_event_received(struct nfc_hci_dev *hdev, u8 pipe, u8 event,
struct sk_buff *skb) struct sk_buff *skb)
{ {
int r = 0; int r = 0;
u8 gate = nfc_hci_pipe2gate(hdev, pipe);
if (gate == 0xff) {
pr_err("Discarded event %x to unopened pipe %x\n", event, pipe);
goto exit;
}
switch (event) { switch (event) {
case NFC_HCI_EVT_TARGET_DISCOVERED: case NFC_HCI_EVT_TARGET_DISCOVERED:
...@@ -307,14 +317,11 @@ void nfc_hci_event_received(struct nfc_hci_dev *hdev, u8 pipe, u8 event, ...@@ -307,14 +317,11 @@ void nfc_hci_event_received(struct nfc_hci_dev *hdev, u8 pipe, u8 event,
goto exit; goto exit;
} }
r = nfc_hci_target_discovered(hdev, r = nfc_hci_target_discovered(hdev, gate);
nfc_hci_pipe2gate(hdev, pipe));
break; break;
default: default:
if (hdev->ops->event_received) { if (hdev->ops->event_received) {
hdev->ops->event_received(hdev, hdev->ops->event_received(hdev, gate, event, skb);
nfc_hci_pipe2gate(hdev, pipe),
event, skb);
return; return;
} }
...@@ -419,6 +426,10 @@ static int hci_dev_version(struct nfc_hci_dev *hdev) ...@@ -419,6 +426,10 @@ static int hci_dev_version(struct nfc_hci_dev *hdev)
r = nfc_hci_get_param(hdev, NFC_HCI_ID_MGMT_GATE, r = nfc_hci_get_param(hdev, NFC_HCI_ID_MGMT_GATE,
NFC_HCI_ID_MGMT_VERSION_SW, &skb); NFC_HCI_ID_MGMT_VERSION_SW, &skb);
if (r == -EOPNOTSUPP) {
pr_info("Software/Hardware info not available\n");
return 0;
}
if (r < 0) if (r < 0)
return r; return r;
......
...@@ -528,6 +528,23 @@ int nfc_llcp_send_i_frame(struct nfc_llcp_sock *sock, ...@@ -528,6 +528,23 @@ int nfc_llcp_send_i_frame(struct nfc_llcp_sock *sock,
if (local == NULL) if (local == NULL)
return -ENODEV; return -ENODEV;
/* Remote is ready but has not acknowledged our frames */
if((sock->remote_ready &&
skb_queue_len(&sock->tx_pending_queue) >= sock->rw &&
skb_queue_len(&sock->tx_queue) >= 2 * sock->rw)) {
pr_err("Pending queue is full %d frames\n",
skb_queue_len(&sock->tx_pending_queue));
return -ENOBUFS;
}
/* Remote is not ready and we've been queueing enough frames */
if ((!sock->remote_ready &&
skb_queue_len(&sock->tx_queue) >= 2 * sock->rw)) {
pr_err("Tx queue is full %d frames\n",
skb_queue_len(&sock->tx_queue));
return -ENOBUFS;
}
msg_data = kzalloc(len, GFP_KERNEL); msg_data = kzalloc(len, GFP_KERNEL);
if (msg_data == NULL) if (msg_data == NULL)
return -ENOMEM; return -ENOMEM;
...@@ -579,7 +596,7 @@ int nfc_llcp_send_ui_frame(struct nfc_llcp_sock *sock, u8 ssap, u8 dsap, ...@@ -579,7 +596,7 @@ int nfc_llcp_send_ui_frame(struct nfc_llcp_sock *sock, u8 ssap, u8 dsap,
struct sk_buff *pdu; struct sk_buff *pdu;
struct nfc_llcp_local *local; struct nfc_llcp_local *local;
size_t frag_len = 0, remaining_len; size_t frag_len = 0, remaining_len;
u8 *msg_ptr; u8 *msg_ptr, *msg_data;
int err; int err;
pr_debug("Send UI frame len %zd\n", len); pr_debug("Send UI frame len %zd\n", len);
...@@ -588,8 +605,17 @@ int nfc_llcp_send_ui_frame(struct nfc_llcp_sock *sock, u8 ssap, u8 dsap, ...@@ -588,8 +605,17 @@ int nfc_llcp_send_ui_frame(struct nfc_llcp_sock *sock, u8 ssap, u8 dsap,
if (local == NULL) if (local == NULL)
return -ENODEV; return -ENODEV;
msg_data = kzalloc(len, GFP_KERNEL);
if (msg_data == NULL)
return -ENOMEM;
if (memcpy_fromiovec(msg_data, msg->msg_iov, len)) {
kfree(msg_data);
return -EFAULT;
}
remaining_len = len; remaining_len = len;
msg_ptr = (u8 *) msg->msg_iov; msg_ptr = msg_data;
while (remaining_len > 0) { while (remaining_len > 0) {
...@@ -616,6 +642,8 @@ int nfc_llcp_send_ui_frame(struct nfc_llcp_sock *sock, u8 ssap, u8 dsap, ...@@ -616,6 +642,8 @@ int nfc_llcp_send_ui_frame(struct nfc_llcp_sock *sock, u8 ssap, u8 dsap,
msg_ptr += frag_len; msg_ptr += frag_len;
} }
kfree(msg_data);
return len; return len;
} }
......
...@@ -656,6 +656,8 @@ static void nfc_llcp_tx_work(struct work_struct *work) ...@@ -656,6 +656,8 @@ static void nfc_llcp_tx_work(struct work_struct *work)
if (llcp_sock == NULL && nfc_llcp_ptype(skb) == LLCP_PDU_I) { if (llcp_sock == NULL && nfc_llcp_ptype(skb) == LLCP_PDU_I) {
nfc_llcp_send_symm(local->dev); nfc_llcp_send_symm(local->dev);
} else { } else {
struct sk_buff *copy_skb = NULL;
u8 ptype = nfc_llcp_ptype(skb);
int ret; int ret;
pr_debug("Sending pending skb\n"); pr_debug("Sending pending skb\n");
...@@ -663,22 +665,29 @@ static void nfc_llcp_tx_work(struct work_struct *work) ...@@ -663,22 +665,29 @@ static void nfc_llcp_tx_work(struct work_struct *work)
DUMP_PREFIX_OFFSET, 16, 1, DUMP_PREFIX_OFFSET, 16, 1,
skb->data, skb->len, true); skb->data, skb->len, true);
if (ptype == LLCP_PDU_I)
copy_skb = skb_copy(skb, GFP_ATOMIC);
nfc_llcp_send_to_raw_sock(local, skb, nfc_llcp_send_to_raw_sock(local, skb,
NFC_LLCP_DIRECTION_TX); NFC_LLCP_DIRECTION_TX);
ret = nfc_data_exchange(local->dev, local->target_idx, ret = nfc_data_exchange(local->dev, local->target_idx,
skb, nfc_llcp_recv, local); skb, nfc_llcp_recv, local);
if (!ret && nfc_llcp_ptype(skb) == LLCP_PDU_I) { if (ret) {
skb = skb_get(skb); kfree_skb(copy_skb);
skb_queue_tail(&llcp_sock->tx_pending_queue, goto out;
skb);
} }
if (ptype == LLCP_PDU_I && copy_skb)
skb_queue_tail(&llcp_sock->tx_pending_queue,
copy_skb);
} }
} else { } else {
nfc_llcp_send_symm(local->dev); nfc_llcp_send_symm(local->dev);
} }
out:
mod_timer(&local->link_timer, mod_timer(&local->link_timer,
jiffies + msecs_to_jiffies(2 * local->remote_lto)); jiffies + msecs_to_jiffies(2 * local->remote_lto));
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment