diff options
author | Marcin Kraglak <marcin.kraglak@tieto.com> | 2014-01-10 10:18:19 +0100 |
---|---|---|
committer | Johan Hedberg <johan.hedberg@intel.com> | 2014-01-16 12:23:15 +0200 |
commit | f167df0e3a3be1a7fb37e1a1bdd8fa457ab77c05 (patch) | |
tree | b4fa720be3654d8b805b31d8630e1dbfe3a41b0a | |
parent | cbc1720d4f13b6d8a25749cdcb4154d4ef08f38b (diff) |
emulator/bthost: Add recv_pn implementation
-rw-r--r-- | emulator/bthost.c | 53 |
1 files changed, 53 insertions, 0 deletions
diff --git a/emulator/bthost.c b/emulator/bthost.c index 5aab60a7f..d0a4ae9ac 100644 --- a/emulator/bthost.c +++ b/emulator/bthost.c @@ -50,6 +50,7 @@ #define RFCOMM_CTRL(type, pf) (((type & 0xef) | (pf << 4))) #define RFCOMM_LEN8(len) (((len) << 1) | 1) #define RFCOMM_LEN16(len) ((len) << 1) +#define RFCOMM_MCC_TYPE(cr, type) (((type << 2) | (cr << 1) | 0x01)) /* RFCOMM FCS calculation */ #define CRC(data) (rfcomm_crc_table[rfcomm_crc_table[0xff ^ data[0]] ^ data[1]]) @@ -101,6 +102,11 @@ static uint8_t rfcomm_fcs2(uint8_t *data) return 0xff - rfcomm_crc_table[CRC(data) ^ data[2]]; } +static uint8_t rfcomm_fcs(uint8_t *data) +{ + return 0xff - CRC(data); +} + struct cmd { struct cmd *next; struct cmd *prev; @@ -1521,9 +1527,56 @@ static void rfcomm_dm_recv(struct bthost *bthost, struct btconn *conn, { } +static void rfcomm_pn_recv(struct bthost *bthost, struct btconn *conn, + struct l2conn *l2conn, uint8_t cr, + const struct rfcomm_pn *pn) +{ + if (cr) { + uint8_t buf[14]; + struct rfcomm_hdr *hdr = (struct rfcomm_hdr *)buf; + struct rfcomm_mcc *mcc = (struct rfcomm_mcc *)(buf + + sizeof(*hdr)); + struct rfcomm_pn *pn_cmd = (struct rfcomm_pn *) + (buf + sizeof(*hdr) + sizeof(*mcc)); + + memset(buf, 0, sizeof(buf)); + + hdr->address = RFCOMM_ADDR(1, 0); + hdr->control = RFCOMM_CTRL(RFCOMM_UIH, 0); + hdr->length = RFCOMM_LEN8(sizeof(*mcc) + sizeof(*pn_cmd)); + + mcc->type = RFCOMM_MCC_TYPE(0, RFCOMM_PN); + mcc->length = RFCOMM_LEN8(sizeof(*pn_cmd)); + + pn_cmd->dlci = pn->dlci; + pn_cmd->priority = pn->priority; + pn_cmd->ack_timer = pn->ack_timer; + pn_cmd->max_retrans = pn->max_retrans; + pn_cmd->mtu = pn->mtu; + pn_cmd->credits = pn->credits; + + buf[sizeof(*hdr) + sizeof(*mcc) + sizeof(*pn_cmd)] = + rfcomm_fcs(buf); + + send_acl(bthost, conn->handle, l2conn->dcid, buf, sizeof(buf)); + } +} + static void rfcomm_mcc_recv(struct bthost *bthost, struct btconn *conn, struct l2conn *l2conn, const void *data, uint16_t len) { + const struct rfcomm_mcc *mcc = data; + uint8_t type = RFCOMM_GET_MCC_TYPE(mcc->type); + + switch (type) { + case RFCOMM_PN: + rfcomm_pn_recv(bthost, conn, l2conn, + RFCOMM_TEST_CR(mcc->type) / 2, + data + sizeof(*mcc)); + break; + default: + break; + } } static void rfcomm_uih_recv(struct bthost *bthost, struct btconn *conn, |